使用dubbo-go搭建dubbo接口测试平台

简介: 作为接口测试平台,没办法引入所有提供方定义的接口jar包,可以有以下方案来解决:dubbo支持telnet协议调用dubbo接口dubbo的泛化调用可以在不引入提供方接口定义jar包的情况下对接口进行调用

背景


http接口测试只需要一个curl命令,但dubbo协议没有这样的现成接口测试工具。通常公司内的dubbo控制台或其他平台会集成一个dubbo接口测试工具。


调用一个dubbo接口,需要知道服务名service、方法名method和参数args


正常的调用,调用方需引入服务提供方定义的接口jar包。


作为接口测试平台,没办法引入所有提供方定义的接口jar包,可以有以下方案来解决:


  1. dubbo支持telnet协议调用dubbo接口
  2. dubbo的泛化调用可以在不引入提供方接口定义jar包的情况下对接口进行调用


对于方案1,实现成本很低,甚至可以在服务器上直接用telnet测试


微信截图_20220423150340.png


它也有缺点


  • 调用无法经过filter
  • 无法携带隐式参数attachment

刚好我们把方案1的优缺点都踩了,我们的dubbo控制台是go语言编写,短时间快速实现,就采用了telnet的方式。


随着业务的发展,流量染色,或标签路由等需要携带隐式参数。


没有走自定义filter,导致业务接口执行不符合预期等都迫使我们升级为泛化调用。


dubbo接口泛化调用在控制台是go编写的情况下也有两个方案可选:


  1. 单独起一个java进程,暴露http端口,与go进程进行交互,泛化调用使用dubbo的java sdk进行编写
  2. 控制台引入dubbo-go,使用dubbo-go进行泛化调用


出于对dubbo java版本的了解,方案1肯定可行,只是架构变得复杂。


而方案2由于dubbo-go还是比较新的项目,并不是很了解,所以不确定其可行性和兼容性,但如果能实现,会大大降低架构的复杂度。


dubbo-go介绍


dubbo-go是dubbo的golang实现版本,它出现的初衷是为了让golang和java的dubbo生态互通。


如今dubbo-go支持provider和consumer端,可以作为一个独立的rpc框架使用,同时社区也是dubbo生态中最火的一个。


如果要说它的意义,我觉得除了和java互通外还有一点非常重要,那就是它能发挥golang协程的巨大作用,这一点可以用在dubbo网关上,如果用dubbo-go实现dubbo网关,就无需纠结线程池、异步等问题。


泛化调用的使用


首先provider端提供一个接口,这个不再赘述,非常简单,接口定义如下


package org.newboo.basic.api;
import org.newboo.basic.model.RpcResult;
import org.newboo.basic.model.User;
public interface MyDemoService {
    RpcResult<String> call(User user);
}
package org.newboo.basic.model;
import java.io.Serializable;
public class User implements Serializable {
    private String uid;
    private String name;
    private String remoteServiceTag;
    ...
}


再来编写java版的泛化调用代码,不引入provider方的jar包:


ReferenceConfig<GenericService> reference = new ReferenceConfig<>();
// ①引用服务名
reference.setInterface("org.newboo.basic.api.MyDemoService");
// ②设置泛化调用标志
reference.setGeneric("true");
DubboBootstrap bootstrap = DubboBootstrap.getInstance();
bootstrap.application(new ApplicationConfig("dubbo-demo-api-consumer"))
        .registry(new RegistryConfig("zookeeper://127.0.0.1:2181"))
        .reference(reference)
        .start();
GenericService genericService = ReferenceConfigCache.getCache().get(reference);
String[] ps = new String[1];
// ③参数类型
ps[0] = "org.newboo.basic.model.User";
Object[] ags = new Object[1];
// ④pojo参数使用map构造
Map<String, String> user = new HashMap<>();
user.put("uid", "1");
user.put("name", "roshi");
user.put("remoteServiceTag", "tag");
ags[0] = user;
// ⑤发起调用
Object res = genericService.$invoke("call", ps, ags);
System.out.println(res);


golang版本


var (
  appName         = "UserConsumer"
  referenceConfig = config.ReferenceConfig{
    InterfaceName: "org.newboo.basic.api.MyDemoService",
    Cluster:       "failover",
    // registry需要配置文件
    Registry:      "demoZk",
    Protocol:      dubbo.DUBBO,
    Generic:       true,
  }
)
func init() {
  referenceConfig.GenericLoad(appName) //appName is the unique identification of RPCService
  time.Sleep(1 * time.Second)
}
// need to setup environment variable "CONF_CONSUMER_FILE_PATH" to "conf/client.yml" before run
func main() {
  call()
}
func call() {
  // 设置attachment
  ctx := context.WithValue(context.TODO(), constant.AttachmentKey, map[string]string{"tag":"test"})
  resp, err := referenceConfig.GetRPCService().(*config.GenericService).Invoke(
    ctx,
    []interface{}{
      "call",
      []string{"org.newboo.basic.model.User"},
      []interface{}{map[string]string{"uid":"111","name":"roshi","remoteServiceTag":"hello"}},
    },
  )
  if err != nil {
    panic(err)
  }
  gxlog.CInfo("success called res: %+v\n", resp)
}


这里我设置了一个attachment,也能正常被provider识别


微信截图_20220423150804.png


泛化调用原理


泛化调用GenericService是dubbo默认提供的一个服务。


其提供了一个名为$invoke的方法,该方法参数有三个,第一个参数是真实要调用的方法名,第二个是参数类型数组,第三个是真实的参数数组,其定义为


public interface GenericService {
    Object $invoke(String method, String[] parameterTypes, Object[] args) throws GenericException;
    ...
}


有了这三个参数,利用反射就能调用到真实的接口了。


java版实现细节


实现这种泛化调用主要涉及到两个filter:

  • consumer端的GenericImplFilter
  • provider端的GenericFilter


consumer端的filter将generic标志设置到attachment中,并封装调用为

GenericService.$invoke


provider端filter判断请求是generic时进行拦截,获取调用方法名、参数、参数值,先序列化为pojo对象,再进行反射调用真实接口。


dubbo-go版细节


与java实现基本一致,其中generic_filter充当consumer端的filter,也是将调用封装为GenericService.$invoke,其中还涉及到一个参数类型的转换,将map转换为dubbo-go-hessian2.Object,这样provider端就可以将其反序列化为Object对象。


与其相关的版本变更如下

  • v1.3.0开始支持泛化调用
  • v1.4.0开始支持用户设置attachement
  • v1.5.1开始支持动态tag路由
  • v1.5.7-rc1修复了直连provider时无法走filter的bug


结论


dubbo-go的泛化调用推荐使用>=v1.5.7-rc1版本,其功能几乎已和java版打平,甚至其实现都与java类似。


使用dubbo-go构建网关、接口测试平台、或者打通golang与java技术生态,不失为一个好的选择。


相关文章
|
21天前
|
Go 开发工具
百炼-千问模型通过openai接口构建assistant 等 go语言
由于阿里百炼平台通义千问大模型没有完善的go语言兼容openapi示例,并且官方答复assistant是不兼容openapi sdk的。 实际使用中发现是能够支持的,所以自己写了一个demo test示例,给大家做一个参考。
|
1月前
|
存储 Rust Go
Go nil 空结构体 空接口有什么区别?
本文介绍了Go语言中的`nil`、空结构体和空接口的区别。`nil`是预定义的零值变量,适用于指针、管道等类型;空结构体大小为0,多个空结构体实例指向同一地址;空接口由`_type`和`data`字段组成,仅当两者均为`nil`时,空接口才为`nil`。
Go nil 空结构体 空接口有什么区别?
|
1月前
|
人工智能 供应链 安全
AI辅助安全测试案例某电商-供应链平台平台安全漏洞
【11月更文挑战第13天】该案例介绍了一家电商供应链平台如何利用AI技术进行全面的安全测试,包括网络、应用和数据安全层面,发现了多个潜在漏洞,并采取了有效的修复措施,提升了平台的整体安全性。
|
1月前
|
数据库连接 Go 数据库
Go语言中的错误注入与防御编程。错误注入通过模拟网络故障、数据库错误等,测试系统稳定性
本文探讨了Go语言中的错误注入与防御编程。错误注入通过模拟网络故障、数据库错误等,测试系统稳定性;防御编程则强调在编码时考虑各种错误情况,确保程序健壮性。文章详细介绍了这两种技术在Go语言中的实现方法及其重要性,旨在提升软件质量和可靠性。
34 1
|
1月前
|
测试技术 Go
go语言中测试工具
【10月更文挑战第22天】
33 4
|
1月前
|
监控 安全 测试技术
构建高效的精准测试平台:设计与实现指南
在软件开发过程中,精准测试是确保产品质量和性能的关键环节。一个精准的测试平台能够自动化测试流程,提高测试效率,缩短测试周期,并提供准确的测试结果。本文将分享如何设计和实现一个精准测试平台,从需求分析到技术选型,再到具体的实现步骤。
143 1
|
2月前
|
人工智能 监控 测试技术
云应用开发平台测试
云应用开发平台测试
79 2
|
1月前
|
监控 安全 测试技术
构建高效精准测试平台:设计与实现全攻略
在软件开发过程中,精准测试是确保产品质量的关键环节。一个高效、精准的测试平台能够自动化测试流程,提高测试覆盖率,缩短测试周期。本文将分享如何设计和实现一个精准测试平台,从需求分析到技术选型,再到具体的实现步骤。
67 0
|
3月前
|
存储 Go
Go to Learn Go之接口
Go to Learn Go之接口
37 7
|
4月前
|
测试技术 Android开发 iOS开发
Appium 是一个开源的自动化测试框架,它支持多种平台和多种编程语言
Appium是一款开源自动化测试框架,支持iOS和Android多平台及多种编程语言。通过WebDriver协议,开发者可编写自动化测试脚本。在iPhone上实现屏幕点击等操作需安装Appium及其依赖,启动服务器,并设置所需的测试环境参数。利用Python等语言编写测试脚本,模拟用户交互行为,最后运行测试脚本来验证应用功能。对于iPhone测试,需准备真实设备或Xcode模拟器。
138 1