gencall.go 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275
  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,
  33. c *ZRpcContext) error {
  34. if !c.Multiple {
  35. return g.genCallInCompatibility(ctx, proto, cfg)
  36. }
  37. return g.genCallGroup(ctx, proto, cfg)
  38. }
  39. func (g *Generator) genCallGroup(ctx DirContext, proto parser.Proto, cfg *conf.Config) error {
  40. dir := ctx.GetCall()
  41. head := util.GetHead(proto.Name)
  42. for _, service := range proto.Service {
  43. childPkg, err := dir.GetChildPackage(service.Name)
  44. if err != nil {
  45. return err
  46. }
  47. callFilename, err := format.FileNamingFormat(cfg.NamingFormat, service.Name)
  48. if err != nil {
  49. return err
  50. }
  51. childDir := filepath.Base(childPkg)
  52. filename := filepath.Join(dir.Filename, childDir, fmt.Sprintf("%s.go", callFilename))
  53. isCallPkgSameToPbPkg := childDir == ctx.GetProtoGo().Filename
  54. isCallPkgSameToGrpcPkg := childDir == ctx.GetProtoGo().Filename
  55. functions, err := g.genFunction(proto.PbPackage, service, isCallPkgSameToGrpcPkg)
  56. if err != nil {
  57. return err
  58. }
  59. iFunctions, err := g.getInterfaceFuncs(proto.PbPackage, service, isCallPkgSameToGrpcPkg)
  60. if err != nil {
  61. return err
  62. }
  63. text, err := pathx.LoadTemplate(category, callTemplateFile, callTemplateText)
  64. if err != nil {
  65. return err
  66. }
  67. alias := collection.NewSet()
  68. if !isCallPkgSameToPbPkg {
  69. for _, item := range proto.Message {
  70. msgName := getMessageName(*item.Message)
  71. alias.AddStr(fmt.Sprintf("%s = %s", parser.CamelCase(msgName),
  72. fmt.Sprintf("%s.%s", proto.PbPackage, parser.CamelCase(msgName))))
  73. }
  74. }
  75. pbPackage := fmt.Sprintf(`"%s"`, ctx.GetPb().Package)
  76. protoGoPackage := fmt.Sprintf(`"%s"`, ctx.GetProtoGo().Package)
  77. if isCallPkgSameToGrpcPkg {
  78. pbPackage = ""
  79. protoGoPackage = ""
  80. }
  81. aliasKeys := alias.KeysStr()
  82. sort.Strings(aliasKeys)
  83. if err = util.With("shared").GoFmt(true).Parse(text).SaveTo(map[string]interface{}{
  84. "name": callFilename,
  85. "alias": strings.Join(aliasKeys, pathx.NL),
  86. "head": head,
  87. "filePackage": dir.Base,
  88. "pbPackage": pbPackage,
  89. "protoGoPackage": protoGoPackage,
  90. "serviceName": stringx.From(service.Name).ToCamel(),
  91. "functions": strings.Join(functions, pathx.NL),
  92. "interface": strings.Join(iFunctions, pathx.NL),
  93. }, filename, true); err != nil {
  94. return err
  95. }
  96. }
  97. return nil
  98. }
  99. func (g *Generator) genCallInCompatibility(ctx DirContext, proto parser.Proto,
  100. cfg *conf.Config) error {
  101. dir := ctx.GetCall()
  102. service := proto.Service[0]
  103. head := util.GetHead(proto.Name)
  104. isCallPkgSameToPbPkg := ctx.GetCall().Filename == ctx.GetPb().Filename
  105. isCallPkgSameToGrpcPkg := ctx.GetCall().Filename == ctx.GetProtoGo().Filename
  106. callFilename, err := format.FileNamingFormat(cfg.NamingFormat, service.Name)
  107. if err != nil {
  108. return err
  109. }
  110. filename := filepath.Join(dir.Filename, fmt.Sprintf("%s.go", callFilename))
  111. functions, err := g.genFunction(proto.PbPackage, service, isCallPkgSameToGrpcPkg)
  112. if err != nil {
  113. return err
  114. }
  115. iFunctions, err := g.getInterfaceFuncs(proto.PbPackage, service, isCallPkgSameToGrpcPkg)
  116. if err != nil {
  117. return err
  118. }
  119. text, err := pathx.LoadTemplate(category, callTemplateFile, callTemplateText)
  120. if err != nil {
  121. return err
  122. }
  123. alias := collection.NewSet()
  124. if !isCallPkgSameToPbPkg {
  125. for _, item := range proto.Message {
  126. msgName := getMessageName(*item.Message)
  127. alias.AddStr(fmt.Sprintf("%s = %s", parser.CamelCase(msgName),
  128. fmt.Sprintf("%s.%s", proto.PbPackage, parser.CamelCase(msgName))))
  129. }
  130. }
  131. pbPackage := fmt.Sprintf(`"%s"`, ctx.GetPb().Package)
  132. protoGoPackage := fmt.Sprintf(`"%s"`, ctx.GetProtoGo().Package)
  133. if isCallPkgSameToGrpcPkg {
  134. pbPackage = ""
  135. protoGoPackage = ""
  136. }
  137. aliasKeys := alias.KeysStr()
  138. sort.Strings(aliasKeys)
  139. return util.With("shared").GoFmt(true).Parse(text).SaveTo(map[string]interface{}{
  140. "name": callFilename,
  141. "alias": strings.Join(aliasKeys, pathx.NL),
  142. "head": head,
  143. "filePackage": dir.Base,
  144. "pbPackage": pbPackage,
  145. "protoGoPackage": protoGoPackage,
  146. "serviceName": stringx.From(service.Name).ToCamel(),
  147. "functions": strings.Join(functions, pathx.NL),
  148. "interface": strings.Join(iFunctions, pathx.NL),
  149. }, filename, true)
  150. }
  151. func getMessageName(msg proto.Message) string {
  152. list := []string{msg.Name}
  153. for {
  154. parent := msg.Parent
  155. if parent == nil {
  156. break
  157. }
  158. parentMsg, ok := parent.(*proto.Message)
  159. if !ok {
  160. break
  161. }
  162. tmp := []string{parentMsg.Name}
  163. list = append(tmp, list...)
  164. msg = *parentMsg
  165. }
  166. return strings.Join(list, "_")
  167. }
  168. func (g *Generator) genFunction(goPackage string, service parser.Service,
  169. isCallPkgSameToGrpcPkg bool) ([]string, error) {
  170. functions := make([]string, 0)
  171. for _, rpc := range service.RPC {
  172. text, err := pathx.LoadTemplate(category, callFunctionTemplateFile, callFunctionTemplate)
  173. if err != nil {
  174. return nil, err
  175. }
  176. comment := parser.GetComment(rpc.Doc())
  177. streamServer := fmt.Sprintf("%s.%s_%s%s", goPackage, parser.CamelCase(service.Name),
  178. parser.CamelCase(rpc.Name), "Client")
  179. if isCallPkgSameToGrpcPkg {
  180. streamServer = fmt.Sprintf("%s_%s%s", parser.CamelCase(service.Name),
  181. parser.CamelCase(rpc.Name), "Client")
  182. }
  183. buffer, err := util.With("sharedFn").Parse(text).Execute(map[string]interface{}{
  184. "serviceName": stringx.From(service.Name).ToCamel(),
  185. "rpcServiceName": parser.CamelCase(service.Name),
  186. "method": parser.CamelCase(rpc.Name),
  187. "package": goPackage,
  188. "pbRequest": parser.CamelCase(rpc.RequestType),
  189. "pbResponse": parser.CamelCase(rpc.ReturnsType),
  190. "hasComment": len(comment) > 0,
  191. "comment": comment,
  192. "hasReq": !rpc.StreamsRequest,
  193. "notStream": !rpc.StreamsRequest && !rpc.StreamsReturns,
  194. "streamBody": streamServer,
  195. "isCallPkgSameToGrpcPkg": isCallPkgSameToGrpcPkg,
  196. })
  197. if err != nil {
  198. return nil, err
  199. }
  200. functions = append(functions, buffer.String())
  201. }
  202. return functions, nil
  203. }
  204. func (g *Generator) getInterfaceFuncs(goPackage string, service parser.Service,
  205. isCallPkgSameToGrpcPkg bool) ([]string, error) {
  206. functions := make([]string, 0)
  207. for _, rpc := range service.RPC {
  208. text, err := pathx.LoadTemplate(category, callInterfaceFunctionTemplateFile,
  209. callInterfaceFunctionTemplate)
  210. if err != nil {
  211. return nil, err
  212. }
  213. comment := parser.GetComment(rpc.Doc())
  214. streamServer := fmt.Sprintf("%s.%s_%s%s", goPackage, parser.CamelCase(service.Name),
  215. parser.CamelCase(rpc.Name), "Client")
  216. if isCallPkgSameToGrpcPkg {
  217. streamServer = fmt.Sprintf("%s_%s%s", parser.CamelCase(service.Name),
  218. parser.CamelCase(rpc.Name), "Client")
  219. }
  220. buffer, err := util.With("interfaceFn").Parse(text).Execute(
  221. map[string]interface{}{
  222. "hasComment": len(comment) > 0,
  223. "comment": comment,
  224. "method": parser.CamelCase(rpc.Name),
  225. "hasReq": !rpc.StreamsRequest,
  226. "pbRequest": parser.CamelCase(rpc.RequestType),
  227. "notStream": !rpc.StreamsRequest && !rpc.StreamsReturns,
  228. "pbResponse": parser.CamelCase(rpc.ReturnsType),
  229. "streamBody": streamServer,
  230. })
  231. if err != nil {
  232. return nil, err
  233. }
  234. functions = append(functions, buffer.String())
  235. }
  236. return functions, nil
  237. }