Tars请求过程与协议分析

我们通过官网的demo来进行分析,安装demo如下代码所示。

// 1.安装tars相关工具
go install github.com/TarsCloud/TarsGo/tars/tools/tarsgo@latest
go install github.com/TarsCloud/TarsGo/tars/tools/tars2go@latest
// 2.初始化生成代码(完成代码初始化)
tarsgo make App Server Servant GoModuleName
tarsgo make TestApp HelloGo SayHello github.com/Tars/test

基本文件结构如下所示:

CREATED HelloGo/SayHello.tars (171 bytes)// 协议文件
CREATED HelloGo/SayHello_imp.go (620 bytes)
CREATED HelloGo/client/client.go (444 bytes)
CREATED HelloGo/config.conf (967 bytes)
CREATED HelloGo/debugtool/dumpstack.go (412 bytes)
CREATED HelloGo/go.mod (37 bytes)
CREATED HelloGo/main.go (517 bytes)
CREATED HelloGo/makefile (193 bytes)
CREATED HelloGo/scripts/makefile.tars.gomod (4181 bytes)
CREATED HelloGo/start.sh (56 bytes)

协议文件名称:SayHello.tars,Tars才用自研的Tars协议和其他的Rpc框架有所不同,不过Tars也可以兼容proto协议。是从生成的.tars.go开始触发,obj.servant.TarsInvoke为入口开始执行,将参数进行组装传递到Invoke,在doInvoke中执行Rpc调用,通过adapterProxy发送Rpc调用,通过send发包给对应的RPC服务。对应的发送流程图如下。

Tars请求过程与协议分析_第1张图片

func (c *AdapterProxy) Send(req *requestf.RequestPacket) error {
	// 添加发送次数
	c.sendAdd()
	// 发送请求打包
	sbuf, err := c.servantProxy.proto.RequestPack(req)
	if err != nil {
		TLOG.Debug("protocol wrong:", req.IRequestId)
		return err
	}
	// 发送请求
	return c.tarsClient.Send(sbuf)
}

其中Send的实现是一个消息队列。

在收到RPC请求的调用后,执行TCPHandler的invoke方法进行请求处理和解析,解析完再通过链接推回去。

rsp := h.ts.invoke(ctx, pkg) // 处理请求

connSt.conn.Write(rsp) // 结果回写

总的来说就是逐字节读取数据,通过特定算法来进行识别标签类型,并传递给对应的解析器进行解析。

func (b *Reader) readHead() (ty, tag byte, err error) {
	data, err := b.buf.ReadByte()
	if err != nil {
		return
	}
	// 识别标签的算法
	ty = data & 0x0f
	tag = (data & 0xf0) >> 4
	return
}

Tars对于数据的参数位于codec文件夹,其中包含了所有的数据类型的解析。我们以字符串解析为例。

1.获取字符串类型,通过Reader的SkipToNoCheck实现,看是string1还是string4。

2.根据类型去执行对应的解析器,分别是两种部分的情况,string1 为单行字符串,string4为多行字符串。

3.获取要获取的字符串的长度,(bReadU32,逐字节读取)。

4.通过b.Next提取对应的字符串,并返回。

func (b *Reader) ReadString(data *string, tag byte, require bool) error {
	have, ty, err := b.SkipToNoCheck(tag, require)
	if err != nil {
		return err
	}
	if !have {
		return nil
	}
	//此处省略了string4的判断,重点是突出处理方案
	if ty == STRING1 {
		var length uint8
		err = bReadU8(b.buf, &length)
		if err != nil {
			return fmt.Errorf("read string1 tag:%d error:%v", tag, err)
		}
		buff := b.Next(int(length))
		*data = string(buff)
	} else {
		return fmt.Errorf("need string, tag:%d, but type is %s", tag, getTypeStr(int(ty)))
	}
	return nil
}

执行完成ReadForm之后,我们就可以得到一个基本的请求结构信息了,比如调用的是哪个,方法是哪个,超时情况,都一一描述清楚。

Tars请求过程与协议分析_第2张图片
最后补充一下:Tars服务之间的互相调用,RPC通信是基于TCP实现的,非HTTP协议。

你可能感兴趣的:(java,github,开发语言)