花椒服务端 gRPC 开发实践

2023-11-01 05:10

本文主要是介绍花椒服务端 gRPC 开发实践,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

背景

在移动端平台开发中,为了增加代码复用,降低开发成本,通常会需要采用跨平台的开发技术,花椒也不例外。本次新的单品开发,由于时间紧,人员有限,经过调研选型,最终确定了 Flutter 方案(具体选型过程不在本文讨论之内)。

为了让客户端更专注业务实现,降低接口联调测试成本,我们选用了 gRPC 方案。gRPC 是一个高性能、通用的开源 RPC 框架,由 Google 开发并基于 HTTP/2 协议标准而设计,基于 ProtoBuf(Protocol Buffers)序列化协议开发,且支持当前主流开发语言。gRPC 通过定义一个服务并指定一个可以远程调用的带有参数和返回类型的的方法,使客户端可以直接调用不同机器上的服务应用的方法,就像是本地对象一样。在服务端,服务实现这个接口并且运行 gRPC 服务处理客户端调用。在客户端,有一个 stub 提供和服务端相同的方法。

gRPC

特点:

  • 基于标准化的 IDL(ProtoBuf)来生成服务器端和客户端代码,支持多种主流开发语言。同时可以更好的支持团队与团队之间的接口设计,开发,测试,协作等。

  • 基于 HTTP/2 设计,支持双向流,多路复用,头部压缩。

  • 支持流式发送和响应,批量传输数据,提升性能。

  • ProtoBuf 序列化数据抓包、调试难度较大。我们使用服务端注入方式提供了用户或设备过滤,请求及返回值日志捕获,并开发对应后台模拟抓包展示。

  • 相比 JSON, 对前端不够友好。gRPC 生态提供了 Gateway 的方式为 gRPC 服务代理出 RESTful 接口。

  • ProtoBuf 提供了非常强的扩展性,可以为 protoc 开发定制插件,从而扩展 proto 文件的功能及描述性。

gRPC-Web

gRPC-Web 为前端浏览器提供了 JavaScript 库用来访问 gRPC 服务,但是需要通过 Envoy 提供代理服务。相比 JSON 的方式对前端有够友好,同时也增加了服务端的部署成本。因此在这次项目中前端未使用 gRPC 服务,而是由 gRPC-Gateway 提供代理的 RESTful 接口。

gRPC-Gateway

grpc-gateway 是 protoc 的一个插件,它能读取 gRPC 的服务定义并生成反向代理服务器,将 RESTful 的 JSON 请求转换为 gRPC 的方式。这样无需太多工作即可实现一套基于 gRPC 服务的 RESTful 接口,方便前端使用调用接口,同时也方便开发过程中通过 Postman/Paw 之类的工具调试接口。

gateway -> gRPC 映射方式:

  • HTTP 源 IP 添加到 gRPC 的 X-Forwarded-For 请求头

  • HTTP 请求 Host 添加到 gRPC 的 X-Forwarded-Host 请求头

  • HTTP 请求头 Authorization 添加到 gRPC 的 authorization 请求头

  • HTTP 请求头带 Grpc-Metadata- 前缀的映射到 gRPC 的 metadata(key 名不带前缀)

例如,gRPC 接口要求的通用的 metadata 参数(如 platform,device_id 等)在 HTTP RESTful 的传递方式如下:

GET /index HTTP/1.1
grpc-metadata-platform: ios
grpc-metadata-device_id: xxxxxxxxx
grpc-metadata-timestamp: 1562641496
grpc-metadata-locale: en_US
grpc-metadata-version: 1.0.0
grpc-metadata-user_id: 
grpc-metadata-user_token: 
Host: gateway.hostame.com

基础库

dart

为了便于客户端调用,连接复用及通用参数传递,我们封装了 dart 的基础库。

BaseClient 维护了针对 HOST 缓存的连接池,同时也提供了接口需要传递的 metadata 信息。

var base = BaseClient(host: 'rpc.hostame.com', port: 443, secure: true);
final md = await base.metadata;
final stub = AuthClient(base.channel, options: CallOptions(metadata: md));

Golang

Golang 后端服务需要同时支持 gRPC 和 Gateway 两种请求方式。为了简化部署和上线依赖,Gateway 和 gRPC 的功能放在了一起,并通过拦截器注入对应的功能,主要包括 gRPC 统计,访问日志,接口鉴权,请求参数校验,gateway JSON 编码等。

svrMux := &ServerMux{...ServeMux: http.NewServeMux(),}svrMux.svr = grpc.NewServer(grpc.UnaryInterceptor(middleware.ChainUnaryServer(recovery.UnaryServerInterceptor(recovery.WithRecoveryHandler(svrMux.recoveryHandler)),prometheus.UnaryServerInterceptor, // prometheus 统计svrMux.UnaryResponseInterceptor(),proto.UnaryServerInterceptor(), // metadata 解析log.UnaryServerInterceptor(conf.GlobalCallback()), // 访问日志auth.UnaryServerInterceptor(conf.GlobalCallback()), // 接口鉴权validator.UnaryServerInterceptor(), // 请求参数校验)),grpc.StreamInterceptor(middleware.ChainStreamServer(...)))svrMux.mux = runtime.NewServeMux(runtime.WithMarshalerOption(runtime.MIMEWildcard, json.Proto), // json 设置runtime.WithProtoErrorHandler(svrMux.protoErrorHandler), // 错误处理runtime.WithStreamErrorHandler(svrMux.streamErrorHandler),)// gRPC gateway muxsvrMux.ServeMux.Handle("/", svrMux.mux)

引用到的 package:

github.com/golang/protobuf
github.com/grpc-ecosystem/go-grpc-middleware
github.com/grpc-ecosystem/go-grpc-prometheus
github.com/grpc-ecosystem/grpc-gateway
github.com/lnnujxxy/protoc-gen-validate
github.com/youlu-cn/grpc-gen
go.uber.org/zap
google.golang.org/grpc

开发流程:

为了提高开发效率,方便维护及模块复用,服务端按功能进行组件化开发。每个组件可以单独运行一个服务,也可以和其它组件共同组成一个服务。每个组件都需要实现 Component 接口:

type Component interface {// 组件名称Name() string// 初始化存储InitStorage() error// 初始化注册 gRPCInitGRPC(svc Service) error// 初始化注册 gatewayInitGateway(svc Service) error// cron 回调StorageCron()
}

对应组件开发完成后,需要开发对应的服务容器,步骤如下。

初始化 base package:

base.Init(context.TODO(), cfg, &global.Callback{Authenticator: &auth.Callback{},LogCapture:    &log.Capture{},
})

如需对外提供服务,需要提供端口及 TLS 证书:

base.DefaultServer.AddPublicServer(rpcPort, gatewayPort, setting.TLSConfig)

组件注册:

base.DefaultServer.RegisterComponent(&user.Component{})
base.DefaultServer.RegisterComponent(&push.Component{})
...

监听服务:

base.DefaultServer.Serve()

接口定义及实现

proto 规范

gRPC 基于标准化的 IDL(ProtoBuf)来生成服务器端和客户端代码,我们决定将所有的接口描述及文档说明都放到 proto 文件中,便于查看及修改。对 proto 的接口描述及注释的规范如下:

// 消息类型注释,支持多行,
// 支持 markdown 语法:
//
// > blockquote
//
// | Syntax | Description |
// | ----------- | ----------- |
// | Header | Title |
// | Paragraph | Text |
message ExampleMessage {uint64 id = 1; // 字段注释,简洁
}// 服务说明,支持 markdown 语法
//
// 1. First item
// 2. Second item
// 3. Third item
//
// ```json
// {
//   "firstName": "John",
//   "lastName": "Smith",
//   "age": 25
// }
// ```
service Example {option (auth.visible) = {scope: PUBLIC_SCOPE // Service scope:仅内网可见或对外可见};// 方法说明,支持 markdown//// - [x] Write the press release// - [ ] Update the website// - [ ] Contact the mediarpc test (ExampleMessage) returns (ExampleMessage) {option (auth.access) = {level: LOW_ACCESS_LEVEL // 接口请求权限};option (google.api.http) = {post: "/example/test"body: "*"};}
}

代码生成

Golang

gengo:@protoc -Iproto \-I${GOPATH}/src/github.com/grpc-ecosystem/grpc-gateway/third_party/googleapis \-I${GOPATH}/src/github.com/lnnujxxy/protoc-gen-validate \-I${GOPATH}/src/github.com/youlu-cn/grpc-gen/protoc-gen-auth \--go_out=plugins=grpc:go/pb \--grpc-gateway_out=logtostderr=true:go/pb \--validate_out="lang=go:go/pb" \--auth_out="lang=go:go/pb" \proto/*.proto

SDK 引入:Golang 使用 go mod 的方式直接引入 pb 生成的 .go 文件。

dart

ifeq ($(shell uname), Darwin)
PROTO_ROOT_DIR = $(shell brew --prefix)/Cellar/protobuf/*
else
PROTO_ROOT_DIR = /usr/local
endifgendart:@protoc --dart_out=dart/front_user/lib/src/generated $(PROTO_ROOT_DIR)/include/google/protobuf/*.proto@protoc -Iproto \-I${GOPATH}/src/github.com/grpc-ecosystem/grpc-gateway/third_party/googleapis \-I${GOPATH}/src/github.com/lnnujxxy/protoc-gen-validate \-I${GOPATH}/src/github.com/youlu-cn/grpc-gen/protoc-gen-auth \--dart_out=grpc:dart/user/lib proto/*.proto

SDK 引入:

修改 pubspec.yaml,执行 flutter packages get 或 flutter packages upgrade。

dependencies:flutter:sdk: flutterprotobuf: ^0.13.4grpc: ^1.0.1user:git:url: git@github.com:project/repo.gitpath: dart/user

已知问题:dart 在对 protobuf 生成的类型做 json 编码时,json 中的 key 是字段号而非名字,导致无法与其它语言交互。

K8s已经成为一线大厂分布式平台的标配技术。你是不是还在惆怅怎么掌握它?来这里,大型互联网公司一线工程师亲授,不来虚的,直接上手实战,3天时间带你搭建K8s平台,快速学会K8s,点击下方图片可了解培训详情。

文档生成

gRPC Gateway 提供了通过 proto 文件生成 swagger API 文档,缺点是只支持 Gateway 的 RESTful 接口,并且默认的展示方式有点不符合我们的常规文档使用方式。

我们基于 protoc 插件开发了 protoc-gen-markdown 工具,可以由 proto 文件生成 markdown 文档,提供 gRPC 接口描述,以及 RESTful 接口描述及 JSON 示例,提供全文目录,支持锚点导航等。生成方式如下:

gendoc:@protoc -Iproto \-I${GOPATH}/src/github.com/grpc-ecosystem/grpc-gateway/third_party/googleapis \-I${GOPATH}/src/github.com/lnnujxxy/protoc-gen-validate \-I${GOPATH}/src/github.com/youlu-cn/grpc-gen/protoc-gen-auth \--markdown_out=":doc" \proto/*.proto

文档会在对应路径生成接口列表 README.md,以及每个 protobuf 对应的接口文档。

调试

传统的 RESTful 接口在调试及问题排查时,可以通过抓包或者 MitM(中间人攻击)的方式,配置也比较容易。而 gRPC 因为使用了 HTTP2 及 protobuf 二进制流,抓包及数据流反解难度相对较高,调试及问题排查时会比较复杂。为了解决这个问题,我们通过服务端注入的方式,配合查询后台过滤对应的请求日志,从而实现如下类似抓包的效果。

后续计划

  • gRPC Streaming

  • 框架层集成全链路 Trace 支持

  • 迭代优化框架,提供对应脚手架,简化新组件/服务创建及开发流程

文章来源:花椒技术,点击查看原文。

Kubernetes入门与进阶实战培训

Kubernetes入门与进阶实战培训将于2020年9月18日在北京开课,3天时间带你系统掌握Kubernetes,学习效果不好可以继续学习。本次培训包括:Docker基础、容器技术、Docker镜像、数据共享与持久化、Docker实践、Kubernetes基础、Pod基础与进阶、常用对象操作、服务发现、Helm、Kubernetes核心组件原理分析、Kubernetes服务质量保证、调度详解与应用场景、网络、基于Kubernetes的CI/CD、基于Kubernetes的配置管理等。点击下方图片或者阅读原文链接查看详情。

这篇关于花椒服务端 gRPC 开发实践的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/320713

相关文章

基于Qt开发一个简单的OFD阅读器

《基于Qt开发一个简单的OFD阅读器》这篇文章主要为大家详细介绍了如何使用Qt框架开发一个功能强大且性能优异的OFD阅读器,文中的示例代码讲解详细,有需要的小伙伴可以参考一下... 目录摘要引言一、OFD文件格式解析二、文档结构解析三、页面渲染四、用户交互五、性能优化六、示例代码七、未来发展方向八、结论摘要

Docker集成CI/CD的项目实践

《Docker集成CI/CD的项目实践》本文主要介绍了Docker集成CI/CD的项目实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录一、引言1.1 什么是 CI/CD?1.2 docker 在 CI/CD 中的作用二、Docke

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

SpringBoot实现websocket服务端及客户端的详细过程

《SpringBoot实现websocket服务端及客户端的详细过程》文章介绍了WebSocket通信过程、服务端和客户端的实现,以及可能遇到的问题及解决方案,感兴趣的朋友一起看看吧... 目录一、WebSocket通信过程二、服务端实现1.pom文件添加依赖2.启用Springboot对WebSocket

C#图表开发之Chart详解

《C#图表开发之Chart详解》C#中的Chart控件用于开发图表功能,具有Series和ChartArea两个重要属性,Series属性是SeriesCollection类型,包含多个Series对... 目录OverviChina编程ewSeries类总结OverviewC#中,开发图表功能的控件是Char

鸿蒙开发搭建flutter适配的开发环境

《鸿蒙开发搭建flutter适配的开发环境》文章详细介绍了在Windows系统上如何创建和运行鸿蒙Flutter项目,包括使用flutterdoctor检测环境、创建项目、编译HAP包以及在真机上运... 目录环境搭建创建运行项目打包项目总结环境搭建1.安装 DevEco Studio NEXT IDE

Python开发围棋游戏的实例代码(实现全部功能)

《Python开发围棋游戏的实例代码(实现全部功能)》围棋是一种古老而复杂的策略棋类游戏,起源于中国,已有超过2500年的历史,本文介绍了如何用Python开发一个简单的围棋游戏,实例代码涵盖了游戏的... 目录1. 围棋游戏概述1.1 游戏规则1.2 游戏设计思路2. 环境准备3. 创建棋盘3.1 棋盘类

基于MySQL Binlog的Elasticsearch数据同步实践

一、为什么要做 随着马蜂窝的逐渐发展,我们的业务数据越来越多,单纯使用 MySQL 已经不能满足我们的数据查询需求,例如对于商品、订单等数据的多维度检索。 使用 Elasticsearch 存储业务数据可以很好的解决我们业务中的搜索需求。而数据进行异构存储后,随之而来的就是数据同步的问题。 二、现有方法及问题 对于数据同步,我们目前的解决方案是建立数据中间表。把需要检索的业务数据,统一放到一张M

这15个Vue指令,让你的项目开发爽到爆

1. V-Hotkey 仓库地址: github.com/Dafrok/v-ho… Demo: 戳这里 https://dafrok.github.io/v-hotkey 安装: npm install --save v-hotkey 这个指令可以给组件绑定一个或多个快捷键。你想要通过按下 Escape 键后隐藏某个组件,按住 Control 和回车键再显示它吗?小菜一碟: <template

Hadoop企业开发案例调优场景

需求 (1)需求:从1G数据中,统计每个单词出现次数。服务器3台,每台配置4G内存,4核CPU,4线程。 (2)需求分析: 1G / 128m = 8个MapTask;1个ReduceTask;1个mrAppMaster 平均每个节点运行10个 / 3台 ≈ 3个任务(4    3    3) HDFS参数调优 (1)修改:hadoop-env.sh export HDFS_NAMENOD