大家好,我是六月天天。如题所述,从今天开始,我将和大家一起逐步完成一个微服务框架。
整个迭代过程会围绕着两个核心思想进行:
- 关注技术选型背后的思想。虽然最终某个技术选型的可能并不是你喜欢的方案(如RPC、日志、数据库等,你可以fork后自行调整),但我们更关注各个技术组件背后的原理与思想,选择的过程比结果更重要;
- 聚焦于简单,关注可维护性。技术框架是项目的基础设施,也是排查复杂业务问题的根本,所以框架层的功能会尽量考虑简单易用,可以让我们花更多的心思在业务开发中。许多开源库提供了大量扩展功能,但我们使用时会尽量克制,减少学习和排查问题时的成本。
微服务框架系列重点介绍框架的搭建过程,期间对一些细节技术点的讲解,会在另一个系列Go语言技巧系列中展开。
v0.1.0:搭建gRPC+HTTP的双重网关服务
项目链接 https://github.com/Junedayday/micro_web_service/tree/v0.1.0
gRPC-gateway官方Github https://github.com/grpc-ecosystem/grpc-gateway
有很多朋友更喜欢使用Gin框架,但我依然选择了gRPC-gateway。
主要在于gRPC-gateway方案对接Google提供的各种开源插件生态都很棒。大家会在后面框架的迭代过程中慢慢体会到它的特性。
后续我也会对Gin做一些分析。
目标
完成RPC服务的框架的搭建
关键技术点
protobuffer
定义IDL(Interface Definition Language 接口定义语言)buf
工具生成Go
代码(包括数据结构和RPC相关服务)Go
项目实现RPC服务(实现RPC接口)
目录构造
1 | --- micro_web_service 项目目录 |
1. protobuffer定义IDL
我们先看一下项目中的demo.proto
文件,重点关注 rpc Echo(DemoRequest) returns (DemoResponse) 这个定义。
1 | message DemoRequest { |
今天我们暂时不对protobuffer
的语法做扩展讲解,只需要简单地了解下它的请求结构体DemoRequest
和响应结构体DemoResponse
。
2. buf工具生成Go代码
我们通过运行项目根目录中的gen.sh
,会在gen
目录下生成对应的Go语言代码。
这部分是自动化的工作,每次修改proto
文件后需要运行。
buf工具的安装请参考README.md,它是protoc的演进版本,不再需要大量flag参数,更加简单易用。
注意,如果修改了模块名,buf工具第一次初始化建议使用 buf beta mod init 指令
3.Go项目实现RPC服务
我们梳理一下整个逻辑,来看看这个Go
程序是怎么提供RPC服务的。
- 在
buf.gen.yaml
中定义了生成的2种服务,go-grpc
和grpc-gateway
,分别表示gRPC
和HTTP
demo.proto
通过脚本,在gen/idl/demo
生成了2个文件,*_grpc.pb.go
和*.pb.gw.go
,分别表示gRPC
和HTTP
- 在
main
函数中注册两个服务,分别为:- gRPC -
demo.RegisterDemoServiceServer(s, &server.Server{})
- HTTP -
demo.RegisterDemoServiceHandlerFromEndpoint(ctx, mux, *grpcServerEndpoint, opts)
- gRPC -
- 在
internal/server/server.go
中,server.Server
需要实现proto
中定义的方法,所以我们加入接口定义demo.UnsafeDemoServiceServer
- 在
internal/server/demo.go
中,实现一个func (s *Server) Echo(ctx context.Context, req *demo.DemoRequest) (*demo.DemoResponse, error)
方法
项目运行
我们用简单的命令来运行,并用RPC访问
1 | 编译并运行 |
项目的私有化
由于本项目只是一个框架,如果你希望修改为个人的项目,主要改动点在两处:
go.mod
里的模块名,以及Go
代码内部的importproto
文件中定义的go_package
建议用编辑工具全量替换
新增接口示例
添加proto定义
1 | message EmptyMessage { |
生成Go文件
1 | bash gen.sh |
添加接口定义
这时候,我们会发现main.go
中有报错,即提示server.Server
这个对象需要实现Empty
方法。于是,我们在internal/server/demo.go
里添加
1 | func (s *Server) Empty(ctx context.Context, req *demo.EmptyMessage) (*demo.EmptyMessage, error) { |
测试新接口
1 | 编译并运行 |
总结
v0.1.0
版本是一个非常简单的web框架,只有样例的RPC接口。
开放HTTP
接口是为了兼容传统方案,而gRPC
则提供了高性能、跨语言的通信方案。从整个实现过程来看,我们只编写了一个具体的实现、也就是Echo
这个方法,就完成了两种通信方式的兼容。
gRPC-Gateway
方案还有很多很棒的特性,我会在后续逐一介绍并引入。
Github: https://github.com/Junedayday/code_reading
Blog: http://junes.tech/
Bilibili: https://space.bilibili.com/293775192
公众号: golangcoding