go-zero/tools/goctl/api/gogen/genroutes.go

218 lines
5.5 KiB
Go
Raw Normal View History

2020-07-29 17:11:41 +08:00
package gogen
import (
"bytes"
"fmt"
"os"
2020-07-29 17:11:41 +08:00
"path"
"sort"
"strings"
"text/template"
2020-08-08 16:40:10 +08:00
"github.com/tal-tech/go-zero/core/collection"
"github.com/tal-tech/go-zero/tools/goctl/api/spec"
apiutil "github.com/tal-tech/go-zero/tools/goctl/api/util"
"github.com/tal-tech/go-zero/tools/goctl/util"
"github.com/tal-tech/go-zero/tools/goctl/vars"
2020-07-29 17:11:41 +08:00
)
const (
routesFilename = "routes.go"
routesTemplate = `// Code generated by goctl. DO NOT EDIT.
2020-07-29 17:11:41 +08:00
package handler
import (
"net/http"
{{.importPackages}}
)
2020-07-31 17:03:19 +08:00
func RegisterHandlers(engine *rest.Server, serverCtx *svc.ServiceContext) {
2020-07-29 17:11:41 +08:00
{{.routesAdditions}}
}
`
routesAdditionTemplate = `
engine.AddRoutes(
{{.routes}} {{.jwt}}{{.signature}}
)
2020-07-29 17:11:41 +08:00
`
)
var mapping = map[string]string{
"delete": "http.MethodDelete",
"get": "http.MethodGet",
"head": "http.MethodHead",
"post": "http.MethodPost",
"put": "http.MethodPut",
"patch": "http.MethodPatch",
2020-07-29 17:11:41 +08:00
}
type (
group struct {
routes []route
jwtEnabled bool
signatureEnabled bool
authName string
middlewares []string
2020-07-29 17:11:41 +08:00
}
route struct {
method string
path string
handler string
}
)
func genRoutes(dir string, api *spec.ApiSpec) error {
2020-07-29 17:11:41 +08:00
var builder strings.Builder
groups, err := getRoutes(api)
if err != nil {
return err
}
gt := template.Must(template.New("groupTemplate").Parse(routesAdditionTemplate))
for _, g := range groups {
var gbuilder strings.Builder
gbuilder.WriteString("[]rest.Route{")
2020-07-29 17:11:41 +08:00
for _, r := range g.routes {
fmt.Fprintf(&gbuilder, `
{
Method: %s,
Path: "%s",
Handler: %s,
},`,
r.method, r.path, r.handler)
}
2020-07-31 17:11:59 +08:00
var jwt string
2020-07-29 17:11:41 +08:00
if g.jwtEnabled {
jwt = fmt.Sprintf("\n rest.WithJwt(serverCtx.Config.%s.AccessSecret),", g.authName)
2020-07-29 17:11:41 +08:00
}
2020-07-31 17:11:59 +08:00
var signature string
2020-07-29 17:11:41 +08:00
if g.signatureEnabled {
signature = fmt.Sprintf("\n rest.WithSignature(serverCtx.Config.%s.Signature),", g.authName)
2020-07-29 17:11:41 +08:00
}
var routes string
if len(g.middlewares) > 0 {
gbuilder.WriteString("\n}...,")
var params = g.middlewares
for i := range params {
params[i] = "serverCtx." + params[i]
}
var middlewareStr = strings.Join(params, ", ")
routes = fmt.Sprintf("rest.WithMiddlewares(\n[]rest.Middleware{ %s }, \n %s \n),",
middlewareStr, strings.TrimSpace(gbuilder.String()))
} else {
gbuilder.WriteString("\n},")
routes = strings.TrimSpace(gbuilder.String())
}
2020-07-29 17:11:41 +08:00
if err := gt.Execute(&builder, map[string]string{
"routes": routes,
2020-07-29 17:11:41 +08:00
"jwt": jwt,
"signature": signature,
}); err != nil {
return err
}
}
parentPkg, err := getParentPackage(dir)
if err != nil {
return err
}
filename := path.Join(dir, handlerDir, routesFilename)
os.Remove(filename)
2020-07-29 17:11:41 +08:00
fp, created, err := apiutil.MaybeCreateFile(dir, handlerDir, routesFilename)
if err != nil {
return err
}
if !created {
return nil
}
defer fp.Close()
t := template.Must(template.New("routesTemplate").Parse(routesTemplate))
buffer := new(bytes.Buffer)
err = t.Execute(buffer, map[string]string{
"importPackages": genRouteImports(parentPkg, api),
"routesAdditions": strings.TrimSpace(builder.String()),
})
if err != nil {
return err
2020-07-29 17:11:41 +08:00
}
2020-07-29 17:11:41 +08:00
formatCode := formatCode(buffer.String())
_, err = fp.WriteString(formatCode)
return err
}
func genRouteImports(parentPkg string, api *spec.ApiSpec) string {
var importSet = collection.NewSet()
2020-08-10 17:26:47 +08:00
importSet.AddStr(fmt.Sprintf("\"%s\"", util.JoinPackages(parentPkg, contextDir)))
2020-07-29 17:11:41 +08:00
for _, group := range api.Service.Groups {
for _, route := range group.Routes {
folder, ok := apiutil.GetAnnotationValue(route.Annotations, "server", groupProperty)
2020-07-29 17:11:41 +08:00
if !ok {
folder, ok = apiutil.GetAnnotationValue(group.Annotations, "server", groupProperty)
2020-07-29 17:11:41 +08:00
if !ok {
continue
}
}
importSet.AddStr(fmt.Sprintf("%s \"%s\"", toPrefix(folder), util.JoinPackages(parentPkg, handlerDir, folder)))
2020-07-29 17:11:41 +08:00
}
}
imports := importSet.KeysStr()
sort.Strings(imports)
2020-08-27 14:40:05 +08:00
projectSection := strings.Join(imports, "\n\t")
depSection := fmt.Sprintf("\"%s/rest\"", vars.ProjectOpenSourceUrl)
return fmt.Sprintf("%s\n\n\t%s", projectSection, depSection)
2020-07-29 17:11:41 +08:00
}
func getRoutes(api *spec.ApiSpec) ([]group, error) {
var routes []group
for _, g := range api.Service.Groups {
var groupedRoutes group
for _, r := range g.Routes {
handler, ok := apiutil.GetAnnotationValue(r.Annotations, "server", "handler")
if !ok {
return nil, fmt.Errorf("missing handler annotation for route %q", r.Path)
}
handler = getHandlerBaseName(handler) + "Handler(serverCtx)"
folder, ok := apiutil.GetAnnotationValue(r.Annotations, "server", groupProperty)
2020-07-29 17:11:41 +08:00
if ok {
handler = toPrefix(folder) + "." + strings.ToUpper(handler[:1]) + handler[1:]
2020-07-29 17:11:41 +08:00
} else {
folder, ok = apiutil.GetAnnotationValue(g.Annotations, "server", groupProperty)
2020-07-29 17:11:41 +08:00
if ok {
handler = toPrefix(folder) + "." + strings.ToUpper(handler[:1]) + handler[1:]
2020-07-29 17:11:41 +08:00
}
}
groupedRoutes.routes = append(groupedRoutes.routes, route{
method: mapping[r.Method],
path: r.Path,
handler: handler,
})
}
if value, ok := apiutil.GetAnnotationValue(g.Annotations, "server", "jwt"); ok {
groupedRoutes.authName = value
groupedRoutes.jwtEnabled = true
}
if value, ok := apiutil.GetAnnotationValue(g.Annotations, "server", "middleware"); ok {
for _, item := range strings.Split(value, ",") {
groupedRoutes.middlewares = append(groupedRoutes.middlewares, item)
}
}
2020-07-29 17:11:41 +08:00
routes = append(routes, groupedRoutes)
}
return routes, nil
}
func toPrefix(folder string) string {
return strings.ReplaceAll(folder, "/", "")
}