三种解决方案
刚开始有两种思路:
一 是用 http 请求的方式,前面加 beanshell 处理器做下转化,后来发现 vars.put() 只能塞字符串类型的变量。而 vars.putObject 能塞进去 byte[],但是在 http body 体中用 ${abc}取用时是变量的地址。但是直接转化为 String 的方式也不对。字符长度会发生变化。
二 直接写 javasampler 方式继承 jmeter 的一种核心包。感觉这个不通用。每个接口都要写一个 jar 包,复用性比较差。不如把数据类型构造的方式写到一个 jar 包里,然后具有想用数据类型构造的方法都可以复用。
最后选择阴差阳错的选择了第三种方式
三、采用 BeanShell Sampler 的方式
思路:直接利用 beanshell 发送 http 请求,将里面的响应码塞到变量里,然后用 BeanShell Assertion 做断言,判断接口调用是否成功了。
BeanShell Sampler 代码:
import com.lala.protobuf.Define;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.impl.client.CloseableHttpClient;
import java.net.URI;
import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.HttpStatus;
import org.apache.http.NameValuePair;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.client.utils.URLEncodedUtils;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.message.BasicNameValuePair;
import org.apache.http.entity.ByteArrayEntity;
String assemblybody = vars.get("body");
byte[] binarybody;
try {
binarybody = Define.toBuildData(assemblybody);
} catch (IOException e) {
e.printStackTrace();
}
String url = vars.get("url");
HttpPost post = new HttpPost(url);
try{
CloseableHttpClient httpClient = HttpClients.createDefault();
post.setHeader("Content-Type","application/x-protobuf");
post.addHeader("Accept", "*/*");
ByteArrayEntity byteArrayEntity = new ByteArrayEntity(binarybody);
post.setEntity(byteArrayEntity);
HttpResponse response = httpClient.execute(post);
InputStream in = response.getEntity().getContent();
BufferedReader br = new BufferedReader(new InputStreamReader(in, "utf-8"));
StringBuilder strber= new StringBuilder();
String line = null;
while((line = br.readLine())!=null){
strber.append(line+'\n');
}
br.close();
in.close();
result = strber.toString();
System.out.println("response.getStatusLine().getStatusCode()"+response.getStatusLine().getStatusCode());
prev.setResponseCode(String.valueOf(response.getStatusLine().getStatusCode()));
vars.put("code",String.valueOf(response.getStatusLine().getStatusCode()));
log.info("Respnse is " + String.valueOf(response.getStatusLine().getStatusCode()));
if(response.getStatusLine().getStatusCode()!=HttpStatus.SC_OK){
result = "服务器异常";
Failure = true;
}
} catch (Exception e){
System.out.println("请求异常");
throw new RuntimeException(e);
} finally{
post.abort();
}
BeanShell Assertion 断言
String code = vars.get("code");
log.info("code:"+code);
if(!code.equals("200")){
Failure = true;
FailureMessage = "ERROR,check error";
}
本来是想用 beanshellsimpler 的响应码做断言,结果发现它只要代码跑通就 200 了。但是代码里的请求却是 406.所以采用了先把响应码塞进变量里,然后再断言变量值。完美解决问题。
还有遗留的几个问题
1、beanshellsimpler 里,怎么指定当前 simpler 的响应码。用 prev.setresponsecode 的方式不好用,也没继续深究了。
2、应该有更简单的方法。
以上的这个解决方案同样适用一些加解密的算法或者一些数据类型构造的接口调用。