gencall.go 5.6 KB

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