gencall.go 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  1. package generator
  2. import (
  3. "fmt"
  4. "path/filepath"
  5. "sort"
  6. "strings"
  7. "github.com/emicklei/proto"
  8. "github.com/zeromicro/go-zero/core/collection"
  9. conf "github.com/zeromicro/go-zero/tools/goctl/config"
  10. "github.com/zeromicro/go-zero/tools/goctl/rpc/parser"
  11. "github.com/zeromicro/go-zero/tools/goctl/util"
  12. "github.com/zeromicro/go-zero/tools/goctl/util/format"
  13. "github.com/zeromicro/go-zero/tools/goctl/util/pathx"
  14. "github.com/zeromicro/go-zero/tools/goctl/util/stringx"
  15. )
  16. const (
  17. callTemplateText = `{{.head}}
  18. package {{.filePackage}}
  19. import (
  20. "context"
  21. {{.pbPackage}}
  22. {{if ne .pbPackage .protoGoPackage}}{{.protoGoPackage}}{{end}}
  23. "github.com/zeromicro/go-zero/zrpc"
  24. "google.golang.org/grpc"
  25. )
  26. type (
  27. {{.alias}}
  28. {{.serviceName}} interface {
  29. {{.interface}}
  30. }
  31. default{{.serviceName}} struct {
  32. cli zrpc.Client
  33. }
  34. )
  35. func New{{.serviceName}}(cli zrpc.Client) {{.serviceName}} {
  36. return &default{{.serviceName}}{
  37. cli: cli,
  38. }
  39. }
  40. {{.functions}}
  41. `
  42. callInterfaceFunctionTemplate = `{{if .hasComment}}{{.comment}}
  43. {{end}}{{.method}}(ctx context.Context{{if .hasReq}}, in *{{.pbRequest}}{{end}}, opts ...grpc.CallOption) ({{if .notStream}}*{{.pbResponse}}, {{else}}{{.streamBody}},{{end}} error)`
  44. callFunctionTemplate = `
  45. {{if .hasComment}}{{.comment}}{{end}}
  46. func (m *default{{.serviceName}}) {{.method}}(ctx context.Context{{if .hasReq}}, in *{{.pbRequest}}{{end}}, opts ...grpc.CallOption) ({{if .notStream}}*{{.pbResponse}}, {{else}}{{.streamBody}},{{end}} error) {
  47. client := {{if .isCallPkgSameToGrpcPkg}}{{else}}{{.package}}.{{end}}New{{.rpcServiceName}}Client(m.cli.Conn())
  48. return client.{{.method}}(ctx{{if .hasReq}}, in{{end}}, opts...)
  49. }
  50. `
  51. )
  52. // GenCall generates the rpc client code, which is the entry point for the rpc service call.
  53. // It is a layer of encapsulation for the rpc client and shields the details in the pb.
  54. func (g *Generator) GenCall(ctx DirContext, proto parser.Proto, cfg *conf.Config) error {
  55. dir := ctx.GetCall()
  56. service := proto.Service
  57. head := util.GetHead(proto.Name)
  58. fmt.Printf(`
  59. call: %s
  60. pb: %s
  61. grpc: %s
  62. `, ctx.GetCall().Filename, ctx.GetPb().Filename, ctx.GetProtoGo().Filename)
  63. isCallPkgSameToPbPkg := ctx.GetCall().Filename == ctx.GetPb().Filename
  64. isCallPkgSameToGrpcPkg := ctx.GetCall().Filename == ctx.GetProtoGo().Filename
  65. callFilename, err := format.FileNamingFormat(cfg.NamingFormat, service.Name)
  66. if err != nil {
  67. return err
  68. }
  69. filename := filepath.Join(dir.Filename, fmt.Sprintf("%s.go", callFilename))
  70. functions, err := g.genFunction(proto.PbPackage, service, isCallPkgSameToGrpcPkg)
  71. if err != nil {
  72. return err
  73. }
  74. iFunctions, err := g.getInterfaceFuncs(proto.PbPackage, service, isCallPkgSameToGrpcPkg)
  75. if err != nil {
  76. return err
  77. }
  78. text, err := pathx.LoadTemplate(category, callTemplateFile, callTemplateText)
  79. if err != nil {
  80. return err
  81. }
  82. alias := collection.NewSet()
  83. if !isCallPkgSameToPbPkg {
  84. for _, item := range proto.Message {
  85. msgName := getMessageName(*item.Message)
  86. alias.AddStr(fmt.Sprintf("%s = %s", parser.CamelCase(msgName), fmt.Sprintf("%s.%s", proto.PbPackage, parser.CamelCase(msgName))))
  87. }
  88. }
  89. pbPackage := fmt.Sprintf(`"%s"`, ctx.GetPb().Package)
  90. protoGoPackage := fmt.Sprintf(`"%s"`, ctx.GetProtoGo().Package)
  91. if isCallPkgSameToGrpcPkg {
  92. pbPackage = ""
  93. protoGoPackage = ""
  94. }
  95. aliasKeys := alias.KeysStr()
  96. sort.Strings(aliasKeys)
  97. err = util.With("shared").GoFmt(true).Parse(text).SaveTo(map[string]interface{}{
  98. "name": callFilename,
  99. "alias": strings.Join(aliasKeys, pathx.NL),
  100. "head": head,
  101. "filePackage": dir.Base,
  102. "pbPackage": pbPackage,
  103. "protoGoPackage": protoGoPackage,
  104. "serviceName": stringx.From(service.Name).ToCamel(),
  105. "functions": strings.Join(functions, pathx.NL),
  106. "interface": strings.Join(iFunctions, pathx.NL),
  107. }, filename, true)
  108. return err
  109. }
  110. func getMessageName(msg proto.Message) string {
  111. list := []string{msg.Name}
  112. for {
  113. parent := msg.Parent
  114. if parent == nil {
  115. break
  116. }
  117. parentMsg, ok := parent.(*proto.Message)
  118. if !ok {
  119. break
  120. }
  121. tmp := []string{parentMsg.Name}
  122. list = append(tmp, list...)
  123. msg = *parentMsg
  124. }
  125. return strings.Join(list, "_")
  126. }
  127. func (g *Generator) genFunction(goPackage string, service parser.Service, isCallPkgSameToGrpcPkg bool) ([]string, error) {
  128. functions := make([]string, 0)
  129. for _, rpc := range service.RPC {
  130. text, err := pathx.LoadTemplate(category, callFunctionTemplateFile, callFunctionTemplate)
  131. if err != nil {
  132. return nil, err
  133. }
  134. comment := parser.GetComment(rpc.Doc())
  135. streamServer := fmt.Sprintf("%s.%s_%s%s", goPackage, parser.CamelCase(service.Name), parser.CamelCase(rpc.Name), "Client")
  136. if isCallPkgSameToGrpcPkg {
  137. streamServer = fmt.Sprintf("%s_%s%s", parser.CamelCase(service.Name), parser.CamelCase(rpc.Name), "Client")
  138. }
  139. buffer, err := util.With("sharedFn").Parse(text).Execute(map[string]interface{}{
  140. "serviceName": stringx.From(service.Name).ToCamel(),
  141. "rpcServiceName": parser.CamelCase(service.Name),
  142. "method": parser.CamelCase(rpc.Name),
  143. "package": goPackage,
  144. "pbRequest": parser.CamelCase(rpc.RequestType),
  145. "pbResponse": parser.CamelCase(rpc.ReturnsType),
  146. "hasComment": len(comment) > 0,
  147. "comment": comment,
  148. "hasReq": !rpc.StreamsRequest,
  149. "notStream": !rpc.StreamsRequest && !rpc.StreamsReturns,
  150. "streamBody": streamServer,
  151. "isCallPkgSameToGrpcPkg": isCallPkgSameToGrpcPkg,
  152. })
  153. if err != nil {
  154. return nil, err
  155. }
  156. functions = append(functions, buffer.String())
  157. }
  158. return functions, nil
  159. }
  160. func (g *Generator) getInterfaceFuncs(goPackage string, service parser.Service, isCallPkgSameToGrpcPkg bool) ([]string, error) {
  161. functions := make([]string, 0)
  162. for _, rpc := range service.RPC {
  163. text, err := pathx.LoadTemplate(category, callInterfaceFunctionTemplateFile, callInterfaceFunctionTemplate)
  164. if err != nil {
  165. return nil, err
  166. }
  167. comment := parser.GetComment(rpc.Doc())
  168. streamServer := fmt.Sprintf("%s.%s_%s%s", goPackage, parser.CamelCase(service.Name), parser.CamelCase(rpc.Name), "Client")
  169. if isCallPkgSameToGrpcPkg {
  170. streamServer = fmt.Sprintf("%s_%s%s", parser.CamelCase(service.Name), parser.CamelCase(rpc.Name), "Client")
  171. }
  172. buffer, err := util.With("interfaceFn").Parse(text).Execute(
  173. map[string]interface{}{
  174. "hasComment": len(comment) > 0,
  175. "comment": comment,
  176. "method": parser.CamelCase(rpc.Name),
  177. "hasReq": !rpc.StreamsRequest,
  178. "pbRequest": parser.CamelCase(rpc.RequestType),
  179. "notStream": !rpc.StreamsRequest && !rpc.StreamsReturns,
  180. "pbResponse": parser.CamelCase(rpc.ReturnsType),
  181. "streamBody": streamServer,
  182. })
  183. if err != nil {
  184. return nil, err
  185. }
  186. functions = append(functions, buffer.String())
  187. }
  188. return functions, nil
  189. }