|
|
|
package gogen
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"path"
|
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
"text/template"
|
|
|
|
|
|
|
|
"zero/tools/goctl/api/spec"
|
|
|
|
apiutil "zero/tools/goctl/api/util"
|
|
|
|
"zero/tools/goctl/util"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
handlerTemplate = `package handler
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"zero/rest/httpx"
|
|
|
|
{{.importPackages}}
|
|
|
|
)
|
|
|
|
|
|
|
|
func {{.handlerName}}(ctx *svc.ServiceContext) http.HandlerFunc {
|
|
|
|
return func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
l := logic.{{.logic}}(r.Context(), ctx)
|
|
|
|
{{.handlerBody}}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`
|
|
|
|
handlerBodyTemplate = `{{.parseRequest}}
|
|
|
|
{{.processBody}}
|
|
|
|
`
|
|
|
|
parseRequestTemplate = `var req {{.requestType}}
|
|
|
|
if err := httpx.Parse(r, &req); err != nil {
|
|
|
|
httpx.Error(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
`
|
|
|
|
hasRespTemplate = `
|
|
|
|
{{.logicResponse}} l.{{.callee}}({{.req}})
|
|
|
|
if err != nil {
|
|
|
|
httpx.Error(w, err)
|
|
|
|
} else {
|
|
|
|
{{.respWriter}}
|
|
|
|
}
|
|
|
|
`
|
|
|
|
)
|
|
|
|
|
|
|
|
func genHandler(dir string, group spec.Group, route spec.Route) error {
|
|
|
|
handler, ok := apiutil.GetAnnotationValue(route.Annotations, "server", "handler")
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("missing handler annotation for %q", route.Path)
|
|
|
|
}
|
|
|
|
handler = getHandlerName(handler)
|
|
|
|
var reqBody string
|
|
|
|
if len(route.RequestType.Name) > 0 {
|
|
|
|
var bodyBuilder strings.Builder
|
|
|
|
t := template.Must(template.New("parseRequest").Parse(parseRequestTemplate))
|
|
|
|
if err := t.Execute(&bodyBuilder, map[string]string{
|
|
|
|
"requestType": typesPacket + "." + util.Title(route.RequestType.Name),
|
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
reqBody = bodyBuilder.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
var req = "req"
|
|
|
|
if len(route.RequestType.Name) == 0 {
|
|
|
|
req = ""
|
|
|
|
}
|
|
|
|
var logicResponse = ""
|
|
|
|
var writeResponse = "nil, nil"
|
|
|
|
var respWriter = `httpx.WriteJson(w, http.StatusOK, resp)`
|
|
|
|
if len(route.ResponseType.Name) > 0 {
|
|
|
|
logicResponse = "resp, err :="
|
|
|
|
writeResponse = "resp, err"
|
|
|
|
} else {
|
|
|
|
logicResponse = "err :="
|
|
|
|
writeResponse = "nil, err"
|
|
|
|
respWriter = `httpx.Ok(w)`
|
|
|
|
}
|
|
|
|
var logicBodyBuilder strings.Builder
|
|
|
|
t := template.Must(template.New("hasRespTemplate").Parse(hasRespTemplate))
|
|
|
|
if err := t.Execute(&logicBodyBuilder, map[string]string{
|
|
|
|
"callee": strings.Title(strings.TrimSuffix(handler, "Handler")),
|
|
|
|
"req": req,
|
|
|
|
"logicResponse": logicResponse,
|
|
|
|
"writeResponse": writeResponse,
|
|
|
|
"respWriter": respWriter,
|
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
respBody := logicBodyBuilder.String()
|
|
|
|
|
|
|
|
if !strings.HasSuffix(handler, "Handler") {
|
|
|
|
handler = handler + "Handler"
|
|
|
|
}
|
|
|
|
|
|
|
|
var bodyBuilder strings.Builder
|
|
|
|
bodyTemplate := template.Must(template.New("handlerBodyTemplate").Parse(handlerBodyTemplate))
|
|
|
|
if err := bodyTemplate.Execute(&bodyBuilder, map[string]string{
|
|
|
|
"parseRequest": reqBody,
|
|
|
|
"processBody": respBody,
|
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return doGenToFile(dir, handler, group, route, bodyBuilder)
|
|
|
|
}
|
|
|
|
|
|
|
|
func doGenToFile(dir, handler string, group spec.Group, route spec.Route, bodyBuilder strings.Builder) error {
|
|
|
|
if getHandlerFolderPath(group, route) != handlerDir {
|
|
|
|
handler = strings.Title(handler)
|
|
|
|
}
|
|
|
|
parentPkg, err := getParentPackage(dir)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
filename := strings.ToLower(handler)
|
|
|
|
if strings.HasSuffix(filename, "handler") {
|
|
|
|
filename = filename + ".go"
|
|
|
|
} else {
|
|
|
|
filename = filename + "handler.go"
|
|
|
|
}
|
|
|
|
fp, created, err := apiutil.MaybeCreateFile(dir, getHandlerFolderPath(group, route), filename)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !created {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
defer fp.Close()
|
|
|
|
t := template.Must(template.New("handlerTemplate").Parse(handlerTemplate))
|
|
|
|
buffer := new(bytes.Buffer)
|
|
|
|
err = t.Execute(buffer, map[string]string{
|
|
|
|
"logic": "New" + strings.TrimSuffix(strings.Title(handler), "Handler") + "Logic",
|
|
|
|
"importPackages": genHandlerImports(group, route, parentPkg),
|
|
|
|
"handlerName": handler,
|
|
|
|
"handlerBody": strings.TrimSpace(bodyBuilder.String()),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
formatCode := formatCode(buffer.String())
|
|
|
|
_, err = fp.WriteString(formatCode)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func genHandlers(dir string, api *spec.ApiSpec) error {
|
|
|
|
for _, group := range api.Service.Groups {
|
|
|
|
for _, route := range group.Routes {
|
|
|
|
if err := genHandler(dir, group, route); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func genHandlerImports(group spec.Group, route spec.Route, parentPkg string) string {
|
|
|
|
var imports []string
|
|
|
|
imports = append(imports, fmt.Sprintf("\"%s\"", path.Join(parentPkg, contextDir)))
|
|
|
|
if len(route.RequestType.Name) > 0 || len(route.ResponseType.Name) > 0 {
|
|
|
|
imports = append(imports, fmt.Sprintf("\"%s\"", path.Join(parentPkg, typesDir)))
|
|
|
|
}
|
|
|
|
imports = append(imports, fmt.Sprintf("\"%s\"", path.Join(parentPkg, getLogicFolderPath(group, route))))
|
|
|
|
sort.Strings(imports)
|
|
|
|
|
|
|
|
return strings.Join(imports, "\n\t")
|
|
|
|
}
|
|
|
|
|
|
|
|
func getHandlerBaseName(handler string) string {
|
|
|
|
handlerName := util.Untitle(handler)
|
|
|
|
if strings.HasSuffix(handlerName, "handler") {
|
|
|
|
handlerName = strings.ReplaceAll(handlerName, "handler", "")
|
|
|
|
} else if strings.HasSuffix(handlerName, "Handler") {
|
|
|
|
handlerName = strings.ReplaceAll(handlerName, "Handler", "")
|
|
|
|
}
|
|
|
|
return handlerName
|
|
|
|
}
|
|
|
|
|
|
|
|
func getHandlerFolderPath(group spec.Group, route spec.Route) string {
|
|
|
|
folder, ok := apiutil.GetAnnotationValue(route.Annotations, "server", folderProperty)
|
|
|
|
if !ok {
|
|
|
|
folder, ok = apiutil.GetAnnotationValue(group.Annotations, "server", folderProperty)
|
|
|
|
if !ok {
|
|
|
|
return handlerDir
|
|
|
|
}
|
|
|
|
}
|
|
|
|
folder = strings.TrimPrefix(folder, "/")
|
|
|
|
folder = strings.TrimSuffix(folder, "/")
|
|
|
|
return path.Join(handlerDir, folder)
|
|
|
|
}
|
|
|
|
|
|
|
|
func getHandlerName(handler string) string {
|
|
|
|
return getHandlerBaseName(handler) + "Handler"
|
|
|
|
}
|