go-zero/tools/goctl/rpc
2020-10-16 10:44:04 +08:00
..
command add in-process cache doc 2020-09-11 15:30:20 +08:00
ctx Goctl rpc patch (#117) 2020-10-10 16:19:46 +08:00
execx Goctl rpc patch (#117) 2020-10-10 16:19:46 +08:00
gen delete goctl rpc main tpl no use import (#130) 2020-10-16 10:44:04 +08:00
parser support api templates 2020-10-15 16:36:49 +08:00
test Goctl rpc patch (#117) 2020-10-10 16:19:46 +08:00
base.pb.go Goctl rpc patch (#117) 2020-10-10 16:19:46 +08:00
base.proto Goctl rpc patch (#117) 2020-10-10 16:19:46 +08:00
CHANGELOG.md Markdown lint (#58) 2020-09-11 19:42:58 +08:00
gen_test.go Goctl rpc patch (#117) 2020-10-10 16:19:46 +08:00
README.md Goctl rpc patch (#117) 2020-10-10 16:19:46 +08:00
test.proto Goctl rpc patch (#117) 2020-10-10 16:19:46 +08:00

Rpc Generation

Goctl Rpc是goctl脚手架下的一个rpc服务代码生成模块支持proto模板生成和rpc服务代码生成通过此工具生成代码你只需要关注业务逻辑编写而不用去编写一些重复性的代码。这使得我们把精力重心放在业务上从而加快了开发效率且降低了代码出错率。

特性

  • 简单易用
  • 快速提升开发效率
  • 出错率低
  • 支持基于main proto作为相对路径的import
  • 支持map、enum类型
  • 支持any类型

快速开始

方式一快速生成greet服务

通过命令 goctl rpc new ${servieName}生成

如生成greet rpc服务

goctl rpc new greet

执行后代码结构如下:

└── greet
  ├── etc
     └── greet.yaml
  ├── go.mod
  ├── go.sum
  ├── greet
     ├── greet.go
     ├── greet_mock.go
     └── types.go
  ├── greet.go
  ├── greet.proto
  ├── internal
     ├── config
        └── config.go
     ├── logic
        └── pinglogic.go
     ├── server
        └── greetserver.go
     └── svc
         └── servicecontext.go
  └── pb
      └── greet.pb.go

rpc一键生成常见问题解决见 常见问题解决

方式二通过指定proto生成rpc服务

  • 生成proto模板

    goctl rpc template -o=user.proto
    
    syntax = "proto3";
    
    package remote;
    
    message Request {
      // 用户名
      string username = 1;
      // 用户密码
      string password = 2;
    }
    
    message Response {
      // 用户名称
      string name = 1;
      // 用户性别
      string gender = 2;
    }
    
    service User {
      // 登录
      rpc Login(Request)returns(Response);
    }
    
  • 生成rpc服务代码

    goctl rpc proto -src=user.proto
    

    代码tree

    user
        ├── etc
        │   └── user.json
        ├── internal
        │   ├── config
        │   │   └── config.go
        │   ├── handler
        │   │   ├── loginhandler.go
        │   ├── logic
        │   │   └── loginlogic.go
        │   └── svc
        │       └── servicecontext.go
        ├── pb
        │   └── user.pb.go
        ├── shared
        │   ├── mockusermodel.go
        │   ├── types.go
        │   └── usermodel.go
        ├── user.go
        └── user.proto
    

准备工作

  • 安装了go环境
  • 安装了protoc&protoc-gen-go并且已经设置环境变量
  • 更多问题请见 注意事项

用法

rpc服务生成用法

goctl rpc proto -h
NAME:
   goctl rpc proto - generate rpc from proto

USAGE:
   goctl rpc proto [command options] [arguments...]

OPTIONS:
   --src value, -s value         the file path of the proto source file
   --dir value, -d value         the target path of the code,default path is "${pwd}". [option]
   --service value, --srv value  the name of rpc service. [option]
   --idea                        whether the command execution environment is from idea plugin. [option]

参数说明

  • --src 必填proto数据源目前暂时支持单个proto文件生成这里不支持不建议外部依赖

  • --dir 非必填默认为proto文件所在目录生成代码的目标目录

  • --service 服务名称非必填默认为proto文件所在目录名称但是如果proto所在目录为一下结构

    user
        ├── cmd
        │   └── rpc
        │       └── user.proto
    

    则服务名称亦为user而非proto所在文件夹名称了这里推荐使用这种结构可以方便在同一个服务名下建立不同类型的服务(api、rpc、mq等),便于代码管理与维护。

    注意这里的shared文件夹名称将会是代码中的package名称。

  • --idea 非必填是否为idea插件中执行保留字段终端执行可以忽略

开发人员需要做什么

关注业务代码编写将重复性、与业务无关的工作交给goctl生成好rpc服务代码后开饭人员仅需要修改

  • 服务中的配置文件编写(etc/xx.json、internal/config/config.go)
  • 服务中业务逻辑编写(internal/logic/xxlogic.go)
  • 服务中资源上下文的编写(internal/svc/servicecontext.go)

注意事项

  • google.golang.org/grpc需要降级到v1.26.0,且protoc-gen-go版本不能高于v1.3.2see https://github.com/grpc/grpc-go/issues/3347)即

    replace google.golang.org/grpc => google.golang.org/grpc v1.26.0
    
  • proto不支持暂多文件同时生成

  • proto不支持外部依赖包引入message不支持inline

  • 目前main文件、shared文件、handler文件会被强制覆盖而和开发人员手动需要编写的则不会覆盖生成这一类在代码头部均有

    // Code generated by goctl. DO NOT EDIT!
    // Source: xxx.proto

的标识,请注意不要将也写业务性代码写在里面。

any和import支持

  • 支持any类型声明

  • 支持import其他proto文件

    any类型固定import为google/protobuf/any.proto,且从${GOPATH}/src中查找proto的import支持main proto的相对路径的import且与proto文件对应的pb.go文件必须在proto目录中能被找到。不支持工程外的其他proto文件import。

⚠️注意: 不支持proto嵌套import被import的proto文件不支持import。

import书写格式

import书写格式

// @{package_of_pb} 
import {proto_omport}

@{package_of_pb}pb文件的真实import目录。 {proto_omport}proto import

demo中的

// @greet/base
import "base/base.proto";

工程目录结构如下

greet
│   ├── base
│   │   ├── base.pb.go
│   │   └── base.proto
│   ├── demo.proto
│   ├── go.mod
│   └── go.sum

demo

syntax = "proto3";
import "google/protobuf/any.proto";
// @greet/base
import "base/base.proto";
package stream;


enum Gender{
  UNKNOWN = 0;
  MAN = 1;
  WOMAN = 2;
}

message StreamResp{
  string name = 2;
  Gender gender = 3;
  google.protobuf.Any details = 5;
  base.StreamReq req = 6;
}
service StreamGreeter {
  rpc greet(base.StreamReq) returns (StreamResp);
}

常见问题解决(go mod工程)

  • 错误一:

    pb/xx.pb.go:220:7: undefined: grpc.ClientConnInterface
    pb/xx.pb.go:224:11: undefined: grpc.SupportPackageIsVersion6
    pb/xx.pb.go:234:5: undefined: grpc.ClientConnInterface
    pb/xx.pb.go:237:24: undefined: grpc.ClientConnInterface
    

    解决方法:请将protoc-gen-go版本降至v1.3.2及一下

  • 错误二:

    
    # go.etcd.io/etcd/clientv3/balancer/picker
    ../../../go/pkg/mod/go.etcd.io/etcd@v0.0.0-20200402134248-51bdeb39e698/clientv3/balancer/picker/err.go:25:9: cannot use &errPicker literal (type *errPicker) as type Picker in return argument:*errPicker does not implement Picker (wrong type for Pick method)
      have Pick(context.Context, balancer.PickInfo) (balancer.SubConn, func(balancer.DoneInfo), error)
      want Pick(balancer.PickInfo) (balancer.PickResult, error)
      ../../../go/pkg/mod/go.etcd.io/etcd@v0.0.0-20200402134248-51bdeb39e698/clientv3/balancer/picker/roundrobin_balanced.go:33:9: cannot use &rrBalanced literal (type *rrBalanced) as type Picker in return argument:
      *rrBalanced does not implement Picker (wrong type for Pick method)
      	have Pick(context.Context, balancer.PickInfo) (balancer.SubConn, func(balancer.DoneInfo), error)
      want Pick(balancer.PickInfo) (balancer.PickResult, error)
      #github.com/tal-tech/go-zero/zrpc/internal/balancer/p2c
      ../../../go/pkg/mod/github.com/tal-tech/go-zero@v1.0.12/zrpc/internal/balancer/p2c/p2c.go:41:32: not enough arguments in call to base.NewBalancerBuilder
      have (string, *p2cPickerBuilder)
    want (string, base.PickerBuilder, base.Config)
    ../../../go/pkg/mod/github.com/tal-tech/go-zero@v1.0.12/zrpc/internal/balancer/p2c/p2c.go:58:9: cannot use &p2cPicker literal (type *p2cPicker) as type balancer.Picker in return argument:
      *p2cPicker does not implement balancer.Picker (wrong type for Pick method)
      	have Pick(context.Context, balancer.PickInfo) (balancer.SubConn, func(balancer.DoneInfo), error)
      	want Pick(balancer.PickInfo) (balancer.PickResult, error)
    

    解决方法:

    replace google.golang.org/grpc => google.golang.org/grpc v1.26.0