概述
- 在一些企业中,各类业务系统非常丰富,相互之间或对外提供很多的服务或接口
- 这些服务或接口中,有很多是需要强契约约束的,服务的提供方、服务的使用方必须遵守相同契约
- 这类服务最典型的就是RPC,其中应用广泛的有Dubbo、gRPC等
- 使用JMeter对这些RPC接口的测试,可以自定义插件来实现
- 由于Dubbo应用相对广泛,本文主要针对Dubbo的接口测试插件开发与实现进行介绍
需要掌握的基础知识
- Java基础技术,像Java基础知识、面向对象、Maven等
- Swing,Java基础的图形化技术,就要用于JMeter组件的界面元件绘制
- JMeter常用组件,包括使用和对应的类、方法,比较典型的配置元件、取样器
- Dubbo,要掌握Dubbo的原理、开发、部署及调用规则,才能较好的进行JMeter插件的开发;尤其是要了解****Dubbo泛化调用的编程。
开发步骤
1、开发配置元件(Config Element)
主要配置Dubbo注册服务地址(一般是Zookepper、Nacos等),超时时间等一些通用配置
继承AbstractConfigGui类,在此类中实现一系列方法,使用Swing绘制界面、并将界面输入框内容保存到JMeter全局数据,具体如下:
getStaticLabel方法,用于返回配置元件名称,示例代码如下:
@Override
public String getStaticLabel() {
return "Dubbo基础配置";
}
getLabelResource方法,获取组件资源名称,用于多语言,可忽略,示例代码如下:
@Override
public String getLabelResource() {
return this.getClass().getSimpleName();
}
configure方法,将测试元件属性值设置回图形化组件,示例代码如下:
@Override
public void configure(TestElement element) {
super.configure(element);
//将配置值设置回当前Swing组件
if (element instanceof ConfigTestElement) {
ConfigTestElement configTestElement = (ConfigTestElement) element;
this.txtDubboAddress.setText(configTestElement.getPropertyAsString(DUBBO_ADDRESS_KEY));
this.txtNamespace.setText(configTestElement.getPropertyAsString(NAMESPACE_KEY));
this.txtGroup.setText(configTestElement.getPropertyAsString(GROUP_KEY));
this.txtProtocol.setText(configTestElement.getPropertyAsString(PROTOCOL_KEY));
this.txtTimeout.setText(configTestElement.getPropertyAsString(TIMEOUT_KEY));
}
}
createTestElement方法,创建测试元件对象,该对象属性值会在测试计划或线程组内共享,示例代码如下:
@Override
public TestElement createTestElement() {
//创建测试元件对象,该对象的属性值会在测试计划或线程组内共享
ConfigTestElement configTestElement = new ConfigTestElement();
//修改测试元件对象
modifyTestElement(configTestElement);
return configTestElement;
}
modifyTestElement方法,修改测试元件对象,主要是将Swing组件输入的值保存到测试元件对象,示例代码如下:
@Override
public void modifyTestElement(TestElement testElement) {
super.configureTestElement(testElement);
//将Swing组件输入的值保存到测试元件
testElement.setProperty(DUBBO_ADDRESS_KEY,this.txtDubboAddress.getText());
testElement.setProperty(NAMESPACE_KEY,this.txtNamespace.getText());
testElement.setProperty(GROUP_KEY,this.txtGroup.getText());
testElement.setProperty(PROTOCOL_KEY,this.txtProtocol.getText());
testElement.setProperty(TIMEOUT_KEY,this.txtTimeout.getText());
}
clearGui方法,清理图形化界面,示例代码如下:
@Override
public void clearGui() {
super.clearGui();
//设置图形化界面组件初始值
this.initGuiValues();
}
2、开发取样器(Sampler)
主要配置单个Dubbo接口测试用例的独特配置,包括接口地址、请求参数、响应数据处理等
分别继承自AbstractSampler、AbstractSamplerGui抽象类,使用Swing绘制界面、并将界面输入框内容保存到JMeter全局数据
重写父类AbstractSampler逻辑和方法
定义与界面相关的取样器数据缓存Key,示例代码如下:
/**
* 服务接口属性Key,用于与DubboSamplerGui共享输入的文本值
*/
public static final String INTERFACE_KEY = "interface_key";
sample方法,执行取样器,示例代码如下
public class DubboSampler extends AbstractSampler{
//1、定义取样器返回对象
SampleResult sampleResult = new SampleResult();
//2、设置取样器标题
sampleResult.setSampleLabel(this.getPropertyAsString("TestElement.name"));
//3、开始取样
sampleResult.sampleStart();
//4、从GUI获取输入值,获取从DubboSamplerGui传递过来的输入值
String inf = this.getPropertyAsString(INTERFACE_KEY);
//...
//5、加工输入数据,使用输入数据构造业务数据
DubboInvokeEntity dubboInvokeEntity = new DubboInvokeEntity();
//...
//6、执行或调用自定义的业务逻辑
Object result = DubboInvokeUtil.invoke(dubboInvokeEntity);
//7、将业务逻辑返回值设置到返回对象
sampleResult.setResponseData(JSON.toJSONString(result), StandardCharsets.UTF_8.name());
//8、结束并返回
sampleResult.setDataType(SampleResult.TEXT);
//结束取样器
sampleResult.sampleEnd();
//标识为成功
sampleResult.setSuccessful(true);
return sampleResult;
}
重写父类AbstractSamplerGui逻辑与方法
定义Swing组件引用,定义与界面相关的Swing组件,可选,示例代码如下:
/**
* 接口名称文本框
*/
private JTextField txtInterface;
//...,多个Swing组件,可进行类似定义
构造方法,进行界面和默认值的初始化,示例代码如下:
public DubboSamplerGui(){
super();
//初始化图形化界面
this.initGui();
//初始化图形化界面默认值
this.initGuiValues();
}
getStaticLabel方法,定义取样器名称,示例代码如下:
@Override
public String getStaticLabel() {
return "Dubbo请求";
}
getLabelResource方法,定义组件资源名称,一般用于多语言资源,使用默认实现,示例代码如下:
@Override
public String getLabelResource() {
return this.getClass().getSimpleName();
}
configure方法,配置方法,将测试元件取样器缓存的属性值设置回图形化组件,示例代码如下:
@Override
public void configure(TestElement element) {
super.configure(element);
if (element instanceof DubboSampler) {
//将Sampler的数据回写GUI,因为一个测试任务下有多个GUI,使用不同的Sampler
DubboSampler sampler = (DubboSampler) element;
this.txtInterface.setText(sampler.getPropertyAsString(DubboSampler.INTERFACE_KEY));
//...
}
}
createTestElement方法,创建测试元件对象,该对象的属性值会在测试计划或线程组内共享,示例代码如下:
@Override
public TestElement createTestElement() {
//创建测试元件对象,该对象的属性值会在测试计划或线程组内共享
DubboSampler dubboSampler = new DubboSampler();
//修改测试元件对象
this.modifyTestElement(dubboSampler);
return dubboSampler;
}
modifyTestElement方法,修改测试元件对象,主要是将Swing组件输入的值保存到测试对象,示例代码如下:
@Override
public void modifyTestElement(TestElement testElement) {
//配置取样器对象
super.configureTestElement(testElement);
if (testElement instanceof DubboSampler) {
DubboSampler dubboSampler = (DubboSampler) testElement;
//将Swing组件输入的值保存到测试元件
dubboSampler.setProperty(DubboSampler.INTERFACE_KEY,this.txtInterface.getText());
//...
}
}
clearGui方法,清理图形化界面,示例代码如下:
@Override
public void clearGui() {
super.clearGui();
//设置图形化界面组件初始值
this.initGuiValues();
}
initGui方法,自定义方法,主要是使用Swing组织取样器的界面元素,示例代码如下:
private void initGui(){
//使用面板等对构造第1步定义的Swing组件,进行合理布局
//...
}
initGuiValues方法,初始化图形化界面Swing输入组件默认值,示例代码如下:
private void initGuiValues() {
this.txtInterface.setText("com.lemon.demo.dubbo.inf.service.IUserService");
//...
}
3、在JMeter中使用
将开发项目打包的.jar包拷贝到JMeter的lib/ext目录
重新启动JMeter
在配置元件、取样器中将会发现Dubb基础配置、Dubbo请求两个插件
效果
总体效果
配置元件效果
取样器效果