proto 语法

概述

Protocol buffers 是 Google 的语言中立、平台中立、可扩展的结构化数据序列化机制——想想 XML,但更小、更快、更简单。您定义了一次数据的结构化方式,然后您可以使用特殊生成的源代码轻松地将结构化数据写入各种数据流并使用各种语言从中读取结构化数据。

proto 语法 - 图1注意

在 go-zero 框架开发中,我们使用 《Language Guide (proto3)》 版本。

快速入门

示例 1. 编写最简单的 rpc 服务

  1. // 声明 proto 语法版本,固定值
  2. syntax = "proto3";
  3. // proto 包名
  4. package greet;
  5. // 生成 golang 代码后的包名
  6. option go_package = "example/proto/greet";
  7. // 定义请求体
  8. message SayHelloReq {}
  9. // 定义响应体
  10. message SayHelloResp {}
  11. // 定义 Greet 服务
  12. service Greet {
  13. // 定义一个 SayHello 一元 rpc 方法,请求体和响应体必填。
  14. rpc SayHello(SayHelloReq) returns (SayHelloResp);
  15. }

示例 2. 编写流式请求服务示例

  1. // 声明 proto 语法版本,固定值
  2. syntax = "proto3";
  3. // proto 包名
  4. package greet;
  5. // 生成 golang 代码后的包名
  6. option go_package = "example/proto/greet";
  7. // 定义结构体
  8. message SayHelloReq {}
  9. message SayHelloResp {}
  10. message SendMessageReq{
  11. string message = 1;
  12. }
  13. message SendMessageResp{
  14. int32 status = 1;
  15. }
  16. message GetMessageReq{
  17. int32 id = 1;
  18. }
  19. message GetMessageResp{
  20. string message = 1;
  21. }
  22. // 定义 Greet 服务
  23. service Greet {
  24. // 定义客户端流式 rpc
  25. rpc SendMessage(stream SendMessageReq) returns (SendMessageResp);
  26. // 定义服务端流式 rpc
  27. rpc GetMessage(GetMessageReq) returns (stream GetMessageResp);
  28. // 定义双向流式 rpc
  29. rpc PushMessage(stream SendMessageReq) returns (stream GetMessageResp);
  30. }

示例 3. 编写 rpc 分组示例

rpc 分组主要通过 service 名称来区分。

  1. // 声明 proto 语法版本,固定值
  2. syntax = "proto3";
  3. // proto 包名
  4. package greet;
  5. // 生成 golang 代码后的包名
  6. option go_package = "example/proto/greet";
  7. // 定义结构体
  8. message SendMessageReq{
  9. string message = 1;
  10. }
  11. message SendMessageResp{
  12. int32 status = 1;
  13. }
  14. message GetMessageReq{
  15. int32 id = 1;
  16. }
  17. message GetMessageResp{
  18. string message = 1;
  19. }
  20. // 定义 Greet 服务
  21. service Greet {
  22. // 定义客户端流式 rpc
  23. rpc SendMessage(stream SendMessageReq) returns (SendMessageResp);
  24. // 定义服务端流式 rpc
  25. rpc GetMessage(GetMessageReq) returns (stream GetMessageResp);
  26. // 定义双向流式 rpc
  27. rpc PushMessage(stream SendMessageReq) returns (stream GetMessageResp);
  28. }
  29. // 定义 Greet 服务
  30. service Greet {
  31. rpc SayHello(SayHelloReq) returns (SayHelloResp);
  32. }
  33. // 定义 Message 服务
  34. service Message {
  35. // 定义客户端流式 rpc
  36. rpc SendMessage(stream SendMessageReq) returns (SendMessageResp);
  37. // 定义服务端流式 rpc
  38. rpc GetMessage(GetMessageReq) returns (stream GetMessageResp);
  39. // 定义双向流式 rpc
  40. rpc PushMessage(stream SendMessageReq) returns (stream GetMessageResp);
  41. }

示例 4. message 示例

  1. // 声明 proto 语法版本,固定值
  2. syntax = "proto3";
  3. // proto 包名
  4. package greet;
  5. // 生成 golang 代码后的包名
  6. option go_package = "example/proto/greet";
  7. // 定义枚举
  8. enum Status{
  9. UNSPECIFIED = 0;
  10. SUCCESS = 1;
  11. FAILED = 2;
  12. }
  13. // 定义结构体
  14. message Base{
  15. int32 code = 1;
  16. string msg = 2;
  17. }
  18. message SendMessageReq{
  19. string message = 1;
  20. }
  21. message SendMessage{
  22. // 使用枚举
  23. Status status = 1;
  24. // 数组
  25. repeated string array = 2;
  26. // map
  27. map<string,int32> map = 3;
  28. // 布尔类型
  29. bool boolean = 4;
  30. // 序列号保留
  31. reserved 5;
  32. }
  33. message SendMessageResp{
  34. Base base = 1;
  35. SendMessage data = 2;
  36. }
  37. // 定义 Greet 服务
  38. service Greet {
  39. // 定义客户端流式 rpc
  40. rpc SendMessage(stream SendMessageReq) returns (SendMessageResp);
  41. }

示例 5. proto 文件引入

假设我们有如下环境:

  1. 工作路径:/usr/local/workspace
  2. base.proto 路径及内容:/usr/local/workspace/base/base.proto
  1. syntax = "proto3";
  2. // proto 包名
  3. package base;
  4. // 生成 golang 代码后的包名
  5. option go_package = "example/proto/base";
  6. message Base{
  7. int32 code = 1;
  8. string msg = 2;
  9. }

现在需要新建 /usr/local/workspace/greet/greet.proto 文件,且需要引用 /usr/local/workspace/base/base.proto 文件中的 Base 结构体,我们来看一下简单的引用示例:

  1. // 声明 proto 语法版本,固定值
  2. syntax = "proto3";
  3. // proto 包名
  4. package greet;
  5. // 生成 golang 代码后的包名
  6. option go_package = "example/proto/greet";
  7. import "base/base.proto";
  8. enum Status{
  9. UNSPECIFIED = 0;
  10. SUCCESS = 1;
  11. FAILED = 2;
  12. }
  13. message SendMessageReq{
  14. string message = 1;
  15. }
  16. message SendMessage{
  17. // 使用枚举
  18. Status status = 1;
  19. // 数组
  20. repeated string array = 2;
  21. // map
  22. map<string,int32> map = 3;
  23. // 布尔类型
  24. bool boolean = 4;
  25. // 序列号保留
  26. reserved 5;
  27. }
  28. message SendMessageResp{
  29. base.Base base = 1;
  30. SendMessage data = 2;
  31. }
  32. // 定义 Greet 服务
  33. service Greet {
  34. // 定义客户端流式 rpc
  35. rpc SendMessage(stream SendMessageReq) returns (SendMessageResp);
  36. }
proto 语法 - 图2温馨提示

goctl 根据 proto 生成 gRPC 代码时:

  1. service 中的 Message(入参&出参) 必须要在 main proto 文件中,不支持引入的文件
  2. 引入的 Message 只能嵌套在 main proto 中的 Message 中使用
  3. goctl 生成 gRPC 代码时,不会生成被引入的 proto 文件的 Go 代码,需要自行预先生成

参考文献