gencall.go 6.2 KB

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