Golang 服务器未正确接收从 Python 客户端发送的 GRPC 请求
知识点掌握了,还需要不断练习才能熟练运用。下面golang学习网给大家带来一个Golang开发实战,手把手教大家学习《Golang 服务器未正确接收从 Python 客户端发送的 GRPC 请求》,在实现功能的过程中也带大家重新温习相关知识点,温故而知新,回头看看说不定又有不一样的感悟!
我编写了一个python grpc客户端,可以连接到许多grpc golang服务。我已经能够像这样完成这项工作:
from alphausblue.connection.conn import grpc_client_connection
from alphausblue.iam.v1.iam_pb2 import whoamirequest
from alphausblue.iam.v1.iam_pb2_grpc import iamstub
async def main():
conn = grpc_client_connection(svc = "blue")
stub = iamstub(conn)
resp = await stub.whoami(whoamirequest())
print(resp)
其中服务名称为 blue。但是,如果我尝试连接到不同的服务来请求如下数据:
from alphausblue.connection.conn import grpc_client_connection
from alphausblue.cost.v1.cost_pb2 import listaccountsrequest
from alphausblue.cost.v1.cost_pb2_grpc import coststub
async def main():
conn = grpc_client_connection(svc = "cost")
stub = coststub(conn)
account = await stub.getaccount(getaccountrequest(vendor = 'aws', id = '731058950257'))
print(account)
我收到未实现的回复。如果该服务不存在但它确实存在并且我的 golang 客户端可以很好地连接到它,那么这是有意义的。此外,当我检查服务器的日志时,我可以清楚地看到请求已到达服务器。做了更多研究,我在我的服务器上发现了这段代码:
type service struct {
userinfo *blueinterceptors.userdata
cost.unimplementedcostserver
}
func (s *service) getaccount(ctx context.context, in *cost.getaccountrequest) (*api.account, error) {
switch in.vendor {
case "aws":
// do stuff
default:
return status.errorf(codes.unimplemented, "not implemented")
}
}
这告诉我,正在调用该函数,但正在反序列化的有效负载缺少 vendor 字段。但是,在调试时我可以看到这一行:
src/core/lib/security/transport/secure_endpoint.cc:296] 写 0000018e2c62fb80: 00 00 00 13 0a 03 61 77 73 12 0c 37 33 31 30 35 38 39 35 30 32 35 37 '......aws ..731058950257'
因此,数据通过 grpc 发送到服务器,但被反序列化为缺少字段的对象。那么这是什么原因呢?
更新
我按照 @blackgreen 的建议查看了 python 和 golang 客户端的 getaccountrequest 的定义。
golang客户端代码:
// request message for the cost.getaccount rpc.
type getaccountrequest struct {
state protoimpl.messagestate
sizecache protoimpl.sizecache
unknownfields protoimpl.unknownfields
vendor string `protobuf:"bytes,1,opt,name=vendor,proto3" json:"vendor,omitempty"`
id string `protobuf:"bytes,2,opt,name=id,proto3" json:"id,omitempty"`
}
python 客户端代码:
_GETACCOUNTREQUEST = _descriptor.Descriptor(
name='GetAccountRequest',
full_name='blueapi.cost.v1.GetAccountRequest',
filename=None,
file=DESCRIPTOR,
containing_type=None,
create_key=_descriptor._internal_create_key,
fields=[
_descriptor.FieldDescriptor(
name='vendor', full_name='blueapi.cost.v1.GetAccountRequest.vendor', index=0,
number=1, type=9, cpp_type=9, label=1,
has_default_value=False, default_value=b"".decode('utf-8'),
message_type=None, enum_type=None, containing_type=None,
is_extension=False, extension_scope=None,
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
_descriptor.FieldDescriptor(
name='id', full_name='blueapi.cost.v1.GetAccountRequest.id', index=1,
number=2, type=9, cpp_type=9, label=1,
has_default_value=False, default_value=b"".decode('utf-8'),
message_type=None, enum_type=None, containing_type=None,
is_extension=False, extension_scope=None,
serialized_options=None, file=DESCRIPTOR, create_key=_descriptor._internal_create_key),
],
extensions=[
],
nested_types=[],
enum_types=[
],
serialized_options=None,
is_extendable=False,
syntax='proto3',
extension_ranges=[],
oneofs=[
],
serialized_start=946,
serialized_end=993,
)
很明显,此处的字段顺序相同,因此我认为这不是问题,除非 grpc 使用 index 而不是 number。
正确答案
确保在 python version 字段中设置的字符串不包含不可见字符后,python 客户端中导入的生成代码与 go 服务器中导入的生成代码可能存在版本不匹配。
特别是,如果两个版本实际上都有 version 字段,但不知何故 go 代码无法“识别”它,则可能是由于字段标记号不匹配造成的。 p>
您显示的字节有效负载(hexa 0a 03 61 77 73 12 0c 37 33 31 30 35 38 39 35 30 32 35 37,base64 cgnhd3msddczmta1odk1mdi1nw==)确实正在发送zq proto 字段中的 bczqbaws 字符串,标签号为 1 ( how do I know)
所以让我们考虑一个人为的示例,python 客户端中使用的 getaccountrequest 的原型架构可能是
message getaccountrequest {
string version = 1; // field tag number 1
}
而go服务器中使用的可能是:
message getaccountrequest {
string version = 3; // field tag number 3
}
在这种情况下,您将在线路上看到具有版本字段的消息,但是在针对具有不同标记号的 go 结构对其进行反序列化时,它将最终变为空。客户端发送带有标签 1 的数据,服务器期望带有标签 3 的数据。
您可以通过检查 go getaccountrequest 结构来验证此假设。它应该如下所示:
type GetAccountRequest struct {
// unexported fields
// other fields, in order
Version string `protobuf:"bytes,3,opt,name=version,proto3" json:"version,omitempty"`
}
反引号`之间的部分是结构体标记,其中3是标记号。
我不知道 python 中的 grpc codegen 是什么样子,但你应该能够以某种方式比较标签号。如果它们确实不同,请确保使用具有相同标记号的生成结构更新客户端或服务器代码。
今天关于《Golang 服务器未正确接收从 Python 客户端发送的 GRPC 请求》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注golang学习网公众号!
比亚迪发布新一代魔方储能系统MC Cube-T:6.432MWh大容量
- 上一篇
- 比亚迪发布新一代魔方储能系统MC Cube-T:6.432MWh大容量
- 下一篇
- Java Servlet中Response对象如何使用
-
- Golang · Go问答 | 1年前 |
- 在读取缓冲通道中的内容之前退出
- 139浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 戈兰岛的全球 GOPRIVATE 设置
- 204浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何将结构作为参数传递给 xml-rpc
- 325浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何用golang获得小数点以下两位长度?
- 478浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何通过 client-go 和 golang 检索 Kubernetes 指标
- 486浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 将多个“参数”映射到单个可变参数的习惯用法
- 439浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 将 HTTP 响应正文写入文件后出现 EOF 错误
- 357浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 结构中映射的匿名列表的“复合文字中缺少类型”
- 352浏览 收藏
-
- Golang · Go问答 | 1年前 |
- NATS Jetstream 的性能
- 101浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何将复杂的字符串输入转换为mapstring?
- 440浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 相当于GoLang中Java将Object作为方法参数传递
- 212浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何确保所有 goroutine 在没有 time.Sleep 的情况下终止?
- 143浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3176次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3388次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3417次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4522次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 3796次使用
-
- GoLand调式动态执行代码
- 2023-01-13 502浏览
-
- 用Nginx反向代理部署go写的网站。
- 2023-01-17 502浏览
-
- Golang取得代码运行时间的问题
- 2023-02-24 501浏览
-
- 请问 go 代码如何实现在代码改动后不需要Ctrl+c,然后重新 go run *.go 文件?
- 2023-01-08 501浏览
-
- 如何从同一个 io.Reader 读取多次
- 2023-04-11 501浏览

