Go语言学习 - RPC篇:gRPC-Gateway示例代码概览

简介: gRPC-Gateway是gRPC生态的一环,用于对HTTP协议的扩展,是一套高性能、高扩展的开源RPC框架。因此,要掌握gRPC-Gateway,必须要对gRPC有一定的基础,才能明白它的定位与价值。

概览

gRPC-Gateway是gRPC生态的一环,用于对HTTP协议的扩展,是一套高性能、高扩展的开源RPC框架。

因此,要掌握gRPC-Gateway,必须要对gRPC有一定的基础,才能明白它的定位与价值。

方案概览

整个方案分为两个方向:

纵向 - RPC协议调用

三个模块:

  • 调用方 - API Client
  • HTTP服务 - Reverse Proxy
  • gRPC服务 - Your gRPC service

两个协议:

  • HTTP - 客户端发起的是HTTP协议,传输到反向代理
  • gRPC - 反向代理与gRPC服务之间的协议是gRPC

关键点:

  1. Reverse Proxy实现了的关键能力是:将HTTP协议转化为gRPC协议
  2. 可同时提供2个服务:HTTP和gRPC,只是HTTP服务的最终实现还是调用到了gRPC

横向 - Protobuf的代码生成

  • gRPC-Gateway部分:自动生成反向代理
  • gRPC部分:自动生成stub

stub这个单词很有意思,相对准确的翻译是存根、残端,和面向对象中的 接口 有异曲同工之妙:提供了实现的框架,但具体实现仍交由开发者

开发工作

对开发者来说,整个方案的工作分为两部分:

  1. RPC部分 - 编写proto文件
  2. 业务逻辑部分 - 编写gRPC中的stub的实现

开发思路

1. RPC开发 - proto文件的编写

我们看一个官方github上的示例proto文件:

syntax = "proto3";
package your.service.v1;
option go_package = "github.com/yourorg/yourprotos/gen/go/your/service/v1";

import "google/api/annotations.proto";

message StringMessage {
  string value = 1;
}

service YourService {
  rpc Echo(StringMessage) returns (StringMessage) {
    option (google.api.http) = {
      post: "/v1/example/echo"
      body: "*"
    };
  }
}

我们从上到下,对里面的语法做简单的分析:

  • proto3版本
  • proto的包名称
  • 生成Go语言代码后的Go Package
  • 导入依赖的proto文件
  • Message - 表示RPC的数据结构,按规则生成到各语言的代码
  • Service - 一组RPC的抽象
  • rpc - 一个具体方法,包括 方法名(请求Message) returns (返回Message)
  • google.api.http - HTTP协议的定义,如示例中的方法和URL

2. 代码生成

如何将proto文件生成为Go语言的stub代码,官方提供了两个路径:

  • buf(新方式,配置简单,推荐)
  • protoc(经典方式,配置较为复杂)

具体的操作方法可以参考: https://github.com/grpc-ecosystem/grpc-gateway#usage

3. 业务实现

代码生成的只是一个stub,具体实现需要我们自己编码。上述方法生成的函数签名大致如下:

func Echo(ctx context.Context, request *proto.StringMessage) returns (response *proto.StringMessage, err error){
   
   
  // 自己实现的业务逻辑
}

关于其中的context与error,我在上一讲已经讲过大致的规范。而在gRPC-Gateway中怎么使用呢?我们会在具体示例中再去讲。

4. main函数框架

1~3步骤将一个RPC请求的开发过程串联了起来,作为web服务的高频迭代部分。

但如果要作为一个完整的服务,还需要包括基础的server启动代码,很少需要改动。我们接着看官方的示例代码:

package main

import (
  "context"
  "flag"
  "net/http"

  "github.com/golang/glog"
  "github.com/grpc-ecosystem/grpc-gateway/v2/runtime"
  "google.golang.org/grpc"
  "google.golang.org/grpc/credentials/insecure"

  gw "github.com/yourorg/yourrepo/proto/gen/go/your/service/v1/your_service"  // Update
)

var (
  // command-line options:
  // gRPC server endpoint
  grpcServerEndpoint = flag.String("grpc-server-endpoint",  "localhost:9090", "gRPC server endpoint")
)

func run() error {
   
   
  ctx := context.Background()
  ctx, cancel := context.WithCancel(ctx)
  defer cancel()

  // Register gRPC server endpoint
  // Note: Make sure the gRPC server is running properly and accessible
  mux := runtime.NewServeMux()
  opts := []grpc.DialOption{
   
   grpc.WithTransportCredentials(insecure.NewCredentials())}
  err := gw.RegisterYourServiceHandlerFromEndpoint(ctx, mux,  *grpcServerEndpoint, opts)
  if err != nil {
   
   
    return err
  }

  // Start HTTP server (and proxy calls to gRPC server endpoint)
  return http.ListenAndServe(":8081", mux)
}

func main() {
   
   
  flag.Parse()
  defer glog.Flush()

  if err := run(); err != nil {
   
   
    glog.Fatal(err)
  }
}

我们要明确一点:这部分代码并没有包括gRPC服务的启动代码,它已经默认在grpcServerEndpoint这个地址+端口上启动了。如果没有进程隔离的强要求,我们可以在main函数中同时启动 gRPC server和gRPC-Gateway server。

gRPC-Gateway启动时有4个重要参数:

  • mux,多路复用,功能最为强大,可以添加各种自定义的拦截器等
  • grpcServerEndpoint,将HTTP转换成gRPC,再发给gRPC的地址
  • opts,服务的启动选项
  • :8081,启动地址

关于gRPC部分内容可以参考: https://grpc.io/docs/languages/go/quickstart/

参考示例

整套框架的完整示例可以参考这个文件 - https://github.com/Junedayday/micro_web_service/blob/master/main.go

结论

本篇重点是对gRPC-Gateway最基础的原理和使用进行了分析。

如果你能通过本篇文章,对这个方案有一个基本认识,那么接下来我会带你玩转这个框架。你也无需担心无法实践到日常项目中:gRPC-Gateway中的能力与gin等框架都是共通的,可以轻松地举一反三。

目录
相关文章
|
3月前
|
程序员 Go 云计算
2023年学习Go语言是否值得?探索Go语言的魅力
2023年学习Go语言是否值得?探索Go语言的魅力
|
3月前
|
缓存 NoSQL Go
通过 SingleFlight 模式学习 Go 并发编程
通过 SingleFlight 模式学习 Go 并发编程
|
9天前
|
数据采集 监控 Java
go语言编程学习
【11月更文挑战第3天】
27 7
|
19天前
|
设计模式 测试技术 Go
学习Go语言
【10月更文挑战第25天】学习Go语言
19 4
|
2月前
|
编译器 Go
go语言学习记录(关于一些奇怪的疑问)有别于其他编程语言
本文探讨了Go语言中的常量概念,特别是特殊常量iota的使用方法及其自动递增特性。同时,文中还提到了在声明常量时,后续常量可沿用前一个值的特点,以及在遍历map时可能遇到的非顺序打印问题。
|
3月前
|
JSON 中间件 Go
go语言后端开发学习(四) —— 在go项目中使用Zap日志库
本文详细介绍了如何在Go项目中集成并配置Zap日志库。首先通过`go get -u go.uber.org/zap`命令安装Zap,接着展示了`Logger`与`Sugared Logger`两种日志记录器的基本用法。随后深入探讨了Zap的高级配置,包括如何将日志输出至文件、调整时间格式、记录调用者信息以及日志分割等。最后,文章演示了如何在gin框架中集成Zap,通过自定义中间件实现了日志记录和异常恢复功能。通过这些步骤,读者可以掌握Zap在实际项目中的应用与定制方法
131 1
go语言后端开发学习(四) —— 在go项目中使用Zap日志库
|
3月前
|
算法 NoSQL 中间件
go语言后端开发学习(六) ——基于雪花算法生成用户ID
本文介绍了分布式ID生成中的Snowflake(雪花)算法。为解决用户ID安全性与唯一性问题,Snowflake算法生成的ID具备全局唯一性、递增性、高可用性和高性能性等特点。64位ID由符号位(固定为0)、41位时间戳、10位标识位(含数据中心与机器ID)及12位序列号组成。面对ID重复风险,可通过预分配、动态或统一分配标识位解决。Go语言实现示例展示了如何使用第三方包`sonyflake`生成ID,确保不同节点产生的ID始终唯一。
go语言后端开发学习(六) ——基于雪花算法生成用户ID
|
3月前
|
Go
Go - 学习 grpc.Dial(target string, opts …DialOption) 的写法
Go - 学习 grpc.Dial(target string, opts …DialOption) 的写法
58 12
|
3月前
|
JSON 缓存 监控
go语言后端开发学习(五)——如何在项目中使用Viper来配置环境
Viper 是一个强大的 Go 语言配置管理库,适用于各类应用,包括 Twelve-Factor Apps。相比仅支持 `.ini` 格式的 `go-ini`,Viper 支持更多配置格式如 JSON、TOML、YAML
go语言后端开发学习(五)——如何在项目中使用Viper来配置环境
|
2月前
|
Rust Linux Go
Rust/Go语言学习
Rust/Go语言学习