Rpc
相关链接
参考示例:
协议
协议用于确定 RPC 中客户端和服务端的消息格式。一般来说,协议都是使用一种与具体的编程语言无关的 IDL ( Interface description language )描述,然后由某种工具转换为各个语言的代码。对于 RPC 来说,这里需要两个步骤:
参考Channel章节中的介绍,开发者需要先利用一些官方的工具为协议文件中的消息类型生成指定编程语言中的代码;
开发者需要使用 AimRT 提供的工具,为协议文件中服务定义生成指定编程语言中的代码;
Protobuf
Protobuf是一种由 Google 开发的、用于序列化结构化数据的轻量级、高效的数据交换格式,是一种广泛使用的 IDL。它不仅能够描述消息结构,还提供了service
语句来定义 RPC 服务。
在使用时,开发者需要先定义一个.proto
文件,在其中定义消息结构和 RPC 服务。例如rpc.proto
:
syntax = "proto3";
package example;
message ExampleReq {
string msg = 1;
int32 num = 2;
}
message ExampleRsp {
uint64 code = 1;
string msg = 2;
}
service ExampleService {
rpc ExampleFunc(ExampleReq) returns (ExampleRsp);
}
然后使用 Protobuf 官方提供的 protoc 工具进行转换,生成消息结构部分的 Python 代码,例如:
protoc --python_out=. rpc.proto
这将生成rpc_pb2.py
文件,包含了根据定义的消息类型生成的 Python 接口。
在这之后,还需要使用 AimRT 提供的 protoc 插件,生成服务定义部分的 Python 桩代码,例如:
protoc --aimrt_rpc_out=. --plugin=protoc-gen-aimrt_rpc=./protoc_plugin_py_gen_aimrt_py_rpc.py rpc.proto
这将生成rpc_aimrt_rpc_pb2.py
文件,包含了根据定义的服务生成的 Python 接口,我们的业务代码中需要 import 此文件。
ROS2 Srv
ROS2 Srv 是一种用于在 ROS2 中进行 RPC 定义的格式。在使用时,开发者需要先定义一个 ROS2 Package,在其中定义一个.srv
文件,比如example.srv
:
byte[] data
---
int64 code
其中,以---
来分割 Req 和 Rsp 的定义。
aimrt_py 安装时会自动安装一个命令行工具 aimrt_py-gen-ros2-rpc
,用于根据 ROS2 Srv 文件生成 AimRT Python 的 RPC 桩代码。
aimrt_py-gen-ros2-rpc --pkg_name=example_pkg --srv_file=./example.srv --output_path=./
其中 pkg_name 是 ROS2 Package 的名称,srv_file 是 ROS2 Srv 文件的路径,output_path 是生成的桩代码的输出路径。这将生成一个example_aimrt_rpc_ros2.py
文件,包含了根据定义的服务生成的 Python 接口,我们的业务代码中需要 import 此文件。
需要注意的是,aimrt_py-gen-ros2-rpc
只会生成 Req 和 Rsp 的定义,不会生成消息结构部分的代码,消息结构部分代码仍然需要使用者自行生成(即构建 ROS2 Package 并生成消息结构代码, 详情可以参考 ROS2 官方文档)。
RpcHandle
模块可以通过调用CoreRef
句柄的GetRpcHandle()
接口,获取RpcHandleRef
句柄。一般情况下,开发者不会直接使用RpcHandleRef
直接提供的接口,而是根据 RPC IDL 文件生成一些桩代码,对RpcHandleRef
句柄做一些封装,然后在业务代码中使用这些经过封装的接口。
这些经过封装的接口的具体形式将在本文档后续章节介绍。开发者在使用 RPC 功能时需要按照以下步骤使用这些接口:
Client 端:
在
Initialize
阶段,调用注册 RPC Client 方法的接口;在
Start
阶段,调用 RPC Invoke 的接口,以实现 RPC 调用;
Server 端:
在
Initialize
阶段,注册 RPC Server 服务的接口;
RpcStatus
在 RPC 调用或者 RPC 处理时,使用者可以通过一个RpcStatus
类型的变量获取 RPC 过程中的错误情况,其包含的接口如下:
OK()->bool
:是否成功;Code()->int
:错误码;ToString()->str
:转字符串;
RpcStatus
类型非常轻量,其中只包含一个错误码字段。使用者可以通过构造函数或 Set 方法设置这个 Code,可以通过 Get 方法获取这个 Code。错误码的枚举值可以参考rpc_status_base.h文件中的定义。
请注意,RpcStatus
中的错误信息一般仅表示框架层面的错误,例如服务未找到、网络错误或者序列化错误等,供开发者排查框架层面的问题。如果开发者需要返回业务层面的错误,建议在业务包中添加相应的字段。
RpcContext
RpcContext 是 RPC 调用时上下文信息,开发者可以在 RPC 调用时设置一些上下文信息,例如超时时间、Meta 信息等,具体接口如下:
CheckUsed()->bool
:检查 Context 是否被使用;SetUsed()->None
:设置 Context 为已使用;Reset()->None
:重置 Context;GetType()->aimrt_rpc_context_type_t
:获取 Context 类型;Timeout()->datetime.timedelta
:获取超时时间;SetTimeout(timeout: datetime.timedelta)->None
:设置超时时间;SetMetaValue(key: str, value: str)->None
:设置元数据;GetMetaValue(key: str)->str
:获取元数据;GetMetaKeys()->List[str]
:获取所有元数据键值对中的键列表;SetToAddr(addr: str)->None
:设置目标地址;GetToAddr()->str
:获取目标地址;SetSerializationType(serialization_type: str)->None
:设置序列化类型;GetSerializationType()->str
:获取序列化类型;GetFunctionName()->str
:获取函数名称;SetFunctionName(func_name: str)->None
:设置函数名称;ToString()->str
:获取上下文信息,以字符串形式返回可读性高的信息;
RpcContextRef
是RpcContext
的引用类型,除不具备Reset
接口外,其他接口与RpcContext
完全相同。
aimrt_rpc_context_type_t
是一个枚举类型,定义了上下文类型,具体值为AIMRT_RPC_CLIENT_CONTEXT
或AIMRT_RPC_SERVER_CONTEXT
,表明这是客户端还是服务端的上下文。
Client
在 AimRT Python RPC 桩代码工具生成的代码里,如xxx_aimrt_rpc_pb2.py
文件里,提供了XXXProxy
类型,开发者基于此 Proxy 接口来发起 RPC 调用。此接口是同步型接口,使用此 Proxy 接口发起 RPC 调用后会阻塞当前线程,直到收到回包或请求超时。
使用该 Proxy 发起 RPC 调用非常简单,一般分为以下几个步骤:
Step 0:引用根据 protobuf 协议生成的桩代码包,例如
xxx_aimrt_rpc_pb2.py
;Step 1:在
Initialize
阶段调用该 Proxy 的RegisterClientFunc
静态方法注册 RPC Client;Step 2:在
Start
阶段里某个业务函数里发起 RPC 调用:Step 2-1:创建一个 Proxy 实例,构造参数是
RpcHandleRef
;Step 2-2:创建 Req,并填充 Req 内容;
Step 2-3:【可选】创建 ctx,设置超时等信息;
Step 2-4:基于 proxy,传入 ctx、Req,发起 RPC 调用,同步等待 RPC 调用结束,保证在整个调用周期里 ctx、Req 都保持有效且不会改动,最终获取返回的 status 和 Rsp;
Step 2-5:解析 status 和 Rsp;
以下是一个使用 AimRT Python 基于 protobuf 协议进行 RPC Client 调用的示例,通过 Create Module 方式拿到CoreRef
句柄。如果是基于Module
模式在Initialize
方法中拿到CoreRef
句柄,使用方式也类似:
import aimrt_py
import threading
import time
import datetime
from google.protobuf.json_format import MessageToJson
import rpc_pb2
import rpc_aimrt_rpc_pb2
def main():
aimrt_core = aimrt_py.Core()
# Initialize
core_options = aimrt_py.CoreOptions()
core_options.cfg_file_path = "path/to/cfg/xxx_cfg.yaml"
aimrt_core.Initialize(core_options)
# Create Module
module_handle = aimrt_core.CreateModule("NormalRpcClientPyModule")
# Register rpc client
rpc_handle = module_handle.GetRpcHandle()
ret = rpc_aimrt_rpc_pb2.ExampleServiceProxy.RegisterClientFunc(rpc_handle)
assert ret, "Register client failed."
# Start
thread = threading.Thread(target=aimrt_core.Start)
thread.start()
# Sleep for seconds
time.sleep(1)
# Call rpc
proxy = rpc_aimrt_rpc_pb2.ExampleServiceProxy(rpc_handle)
req = rpc_pb2.GetFooDataReq()
req.msg = "example msg"
ctx = aimrt_py.RpcContext()
ctx.SetTimeout(datetime.timedelta(seconds=30))
ctx.SetMetaValue("key1", "value1")
status, rsp = proxy.GetFooData(ctx, req)
aimrt_py.info(module_handle.GetLogger(),
f"Call rpc done, "
f"status: {status.ToString()}, "
f"req: {MessageToJson(req)}, "
f"rsp: {MessageToJson(rsp)}")
# Shutdown
aimrt_core.Shutdown()
thread.join()
if __name__ == '__main__':
main()
Server
在 AimRT Python RPC 桩代码工具生成的代码里,如xxx_aimrt_rpc_pb2.py
文件里,提供了一个继承了aimrt_py.ServiceBase
的 Service 基类,开发者需要继承该 Service 基类并实现其中的虚接口。此 Service 接口是同步型接口,开发者只能在 handle 中阻塞的完成所有操作并在最后返回回包。
使用该接口提供 RPC 服务,一般分为以下几个步骤:
Step 0:引用根据 protobuf 协议生成的桩代码包,例如
xxx_aimrt_rpc_pb2.py
;Step 1:开发者实现一个 Impl 类,继承包中的
XXXService
,并实现其中的虚接口,接口形式为(ctx, req)->status, rsp
;Step 1-1:解析 Ctx 和 Req,并填充 Rsp;
Step 1-2:返回
RpcStatus
和 Rsp;
Step 2:在
Initialize
阶段调用RpcHandleRef
的RegisterService
方法注册 RPC Service;
以下是一个使用 AimRT Python 基于 protobuf 协议进行 RPC Service 处理的示例,通过 Create Module 方式拿到CoreRef
句柄。如果是基于Module
模式在Initialize
方法中拿到CoreRef
句柄,使用方式也类似:
import aimrt_py
import threading
import signal
from google.protobuf.json_format import MessageToJson
import rpc_pb2
import rpc_aimrt_rpc_pb2
global_aimrt_core = None
def signal_handler(sig, frame):
global global_aimrt_core
if (global_aimrt_core and (sig == signal.SIGINT or sig == signal.SIGTERM)):
global_aimrt_core.Shutdown()
return
sys.exit(0)
class ExampleServiceImpl(rpc_aimrt_rpc_pb2.ExampleService):
def __init__(self, logger):
super().__init__()
self.logger = logger
@staticmethod
def PrintMetaInfo(logger, ctx_ref):
meta_keys = ctx_ref.GetMetaKeys()
for key in meta_keys:
aimrt_py.info(logger, f"meta key: {key}, value: {ctx_ref.GetMetaValue(key)}")
def GetFooData(self, ctx_ref, req):
rsp = rpc_pb2.GetFooDataRsp()
rsp.msg = "echo " + req.msg
ExampleServiceImpl.PrintMetaInfo(self.logger, ctx_ref)
aimrt_py.info(self.logger,
f"Server handle new rpc call. "
f"context: {ctx_ref.ToString()}, "
f"req: {MessageToJson(req)}, "
f"return rsp: {MessageToJson(rsp)}")
return aimrt_py.RpcStatus(), rsp
def main():
signal.signal(signal.SIGINT, signal_handler)
signal.signal(signal.SIGTERM, signal_handler)
aimrt_core = aimrt_py.Core()
global global_aimrt_core
global_aimrt_core = aimrt_core
# Initialize
core_options = aimrt_py.CoreOptions()
core_options.cfg_file_path = "path/to/cfg/xxx_cfg.yaml"
aimrt_core.Initialize(core_options)
# Create Module
module_handle = aimrt_core.CreateModule("NormalRpcServerPymodule")
# Register rpc service
service = ExampleServiceImpl(module_handle.GetLogger())
ret = module_handle.GetRpcHandle().RegisterService(service)
assert ret, "Register service failed."
# Start
thread = threading.Thread(target=aimrt_core.Start)
thread.start()
while thread.is_alive():
thread.join(1.0)
if __name__ == '__main__':
main()
基于 ROS2 Srv 协议的 RPC 调用和处理,除数据类型不同外,其他使用方式与基于 protobuf 协议的 RPC 调用和处理基本一致。
完整示例请参考: