package gogen import ( "bytes" "fmt" "path" "strings" "text/template" "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" ) const ( handlerTemplate = `package handler import ( "net/http" {{.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 string var writeResponse string 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\"", util.JoinPackages(parentPkg, getLogicFolderPath(group, route)))) imports = append(imports, fmt.Sprintf("\"%s\"", util.JoinPackages(parentPkg, contextDir))) if len(route.RequestType.Name) > 0 || len(route.ResponseType.Name) > 0 { imports = append(imports, fmt.Sprintf("\"%s\"\n", util.JoinPackages(parentPkg, typesDir))) } imports = append(imports, fmt.Sprintf("\"%s/rest/httpx\"", vars.ProjectOpenSourceUrl)) 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" }