package javagen import ( "bufio" "bytes" "errors" "fmt" "io" "path" "strings" "text/template" "github.com/tal-tech/go-zero/core/stringx" "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" ) const ( componentTemplate = `// Code generated by goctl. DO NOT EDIT. package com.xhb.logic.http.packet.{{.packet}}.model; import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.Nullable; {{.imports}} public class {{.className}} extends {{.superClassName}} { {{.properties}} {{if .HasProperty}} public {{.className}}() { } public {{.className}}({{.params}}) { {{.constructorSetter}} } {{end}} {{.getSet}} } ` getSetTemplate = ` {{.indent}}{{.decorator}} {{.indent}}public {{.returnType}} get{{.property}}() { {{.indent}} return this.{{.tagValue}}; {{.indent}}} {{.indent}}public void set{{.property}}({{.type}} {{.propertyValue}}) { {{.indent}} this.{{.tagValue}} = {{.propertyValue}}; {{.indent}}} ` boolTemplate = ` {{.indent}}{{.decorator}} {{.indent}}public {{.returnType}} is{{.property}}() { {{.indent}} return this.{{.tagValue}}; {{.indent}}} {{.indent}}public void set{{.property}}({{.type}} {{.propertyValue}}) { {{.indent}} this.{{.tagValue}} = {{.propertyValue}}; {{.indent}}} ` httpResponseData = "import com.xhb.core.response.HttpResponseData;" httpData = "import com.xhb.core.packet.HttpData;" ) type componentsContext struct { api *spec.ApiSpec requestTypes []spec.Type responseTypes []spec.Type imports []string members []spec.Member } func genComponents(dir, packetName string, api *spec.ApiSpec) error { types := api.Types if len(types) == 0 { return nil } var requestTypes []spec.Type var responseTypes []spec.Type for _, group := range api.Service.Groups { for _, route := range group.Routes { if route.RequestType != nil { requestTypes = append(requestTypes, route.RequestType) } if route.ResponseType != nil { responseTypes = append(responseTypes, route.ResponseType) } } } context := componentsContext{api: api, requestTypes: requestTypes, responseTypes: responseTypes} for _, ty := range types { if err := context.createComponent(dir, packetName, ty); err != nil { return err } } return nil } func (c *componentsContext) createComponent(dir, packetName string, ty spec.Type) error { defineStruct, ok := ty.(spec.DefineStruct) if !ok { return errors.New("unsupported type %s" + ty.Name()) } for _, item := range c.requestTypes { if item.Name() == defineStruct.Name() { if len(defineStruct.GetFormMembers())+len(defineStruct.GetBodyMembers()) == 0 { return nil } } } modelFile := util.Title(ty.Name()) + ".java" filename := path.Join(dir, modelDir, modelFile) if err := util.RemoveOrQuit(filename); err != nil { return err } fp, created, err := apiutil.MaybeCreateFile(dir, modelDir, modelFile) if err != nil { return err } if !created { return nil } defer fp.Close() propertiesString, err := c.buildProperties(defineStruct) if err != nil { return err } getSetString, err := c.buildGetterSetter(defineStruct) if err != nil { return err } superClassName := "HttpData" for _, item := range c.responseTypes { if item.Name() == defineStruct.Name() { superClassName = "HttpResponseData" if !stringx.Contains(c.imports, httpResponseData) { c.imports = append(c.imports, httpResponseData) } break } } if superClassName == "HttpData" && !stringx.Contains(c.imports, httpData) { c.imports = append(c.imports, httpData) } params, constructorSetter, err := c.buildConstructor() if err != nil { return err } buffer := new(bytes.Buffer) t := template.Must(template.New("componentType").Parse(componentTemplate)) err = t.Execute(buffer, map[string]interface{}{ "properties": propertiesString, "params": params, "constructorSetter": constructorSetter, "getSet": getSetString, "packet": packetName, "imports": strings.Join(c.imports, "\n"), "className": util.Title(defineStruct.Name()), "superClassName": superClassName, "HasProperty": len(strings.TrimSpace(propertiesString)) > 0, }) if err != nil { return err } _, err = fp.WriteString(formatSource(buffer.String())) return err } func (c *componentsContext) buildProperties(defineStruct spec.DefineStruct) (string, error) { var builder strings.Builder if err := c.writeType(&builder, defineStruct); err != nil { return "", apiutil.WrapErr(err, "Type "+defineStruct.Name()+" generate error") } return builder.String(), nil } func (c *componentsContext) buildGetterSetter(defineStruct spec.DefineStruct) (string, error) { var builder strings.Builder if err := c.genGetSet(&builder, defineStruct, 1); err != nil { return "", apiutil.WrapErr(err, "Type "+defineStruct.Name()+" get or set generate error") } return builder.String(), nil } func (c *componentsContext) writeType(writer io.Writer, defineStruct spec.DefineStruct) error { c.members = make([]spec.Member, 0) err := c.writeMembers(writer, defineStruct, 1) if err != nil { return err } return nil } func (c *componentsContext) writeMembers(writer io.Writer, defineStruct spec.DefineStruct, indent int) error { for _, member := range defineStruct.Members { if member.IsInline { defineStruct, ok := member.Type.(spec.DefineStruct) if ok { err := c.writeMembers(writer, defineStruct, indent) if err != nil { return err } continue } return errors.New("unsupported inline type %s" + member.Type.Name()) } if member.IsBodyMember() || member.IsFormMember() { if err := writeProperty(writer, member, indent); err != nil { return err } c.members = append(c.members, member) } } return nil } func (c *componentsContext) buildConstructor() (string, string, error) { var params strings.Builder var constructorSetter strings.Builder for index, member := range c.members { tp, err := specTypeToJava(member.Type) if err != nil { return "", "", err } params.WriteString(fmt.Sprintf("%s %s", tp, util.Untitle(member.Name))) pn, err := member.GetPropertyName() if err != nil { return "", "", err } if index != len(c.members)-1 { params.WriteString(", ") } writeIndent(&constructorSetter, 2) constructorSetter.WriteString(fmt.Sprintf("this.%s = %s;", pn, util.Untitle(member.Name))) if index != len(c.members)-1 { constructorSetter.WriteString(util.NL) } } return params.String(), constructorSetter.String(), nil } func (c *componentsContext) genGetSet(writer io.Writer, defineStruct spec.DefineStruct, indent int) error { var members = defineStruct.GetBodyMembers() members = append(members, defineStruct.GetFormMembers()...) for _, member := range members { javaType, err := specTypeToJava(member.Type) if err != nil { return nil } var property = util.Title(member.Name) var templateStr = getSetTemplate if javaType == "boolean" { templateStr = boolTemplate property = strings.TrimPrefix(property, "Is") property = strings.TrimPrefix(property, "is") } t := template.Must(template.New(templateStr).Parse(getSetTemplate)) var tmplBytes bytes.Buffer tyString := javaType decorator := "" javaPrimitiveType := []string{"int", "long", "boolean", "float", "double", "short"} if !stringx.Contains(javaPrimitiveType, javaType) { if member.IsOptional() || member.IsOmitEmpty() { decorator = "@Nullable " } else { decorator = "@NotNull " } tyString = decorator + tyString } tagName, err := member.GetPropertyName() if err != nil { return err } err = t.Execute(&tmplBytes, map[string]string{ "property": property, "propertyValue": util.Untitle(member.Name), "tagValue": tagName, "type": tyString, "decorator": decorator, "returnType": javaType, "indent": indentString(indent), }) if err != nil { return err } r := tmplBytes.String() r = strings.Replace(r, " boolean get", " boolean is", 1) writer.Write([]byte(r)) } return nil } func formatSource(source string) string { var builder strings.Builder scanner := bufio.NewScanner(strings.NewReader(source)) preIsBreakLine := false for scanner.Scan() { text := strings.TrimSpace(scanner.Text()) if text == "" && preIsBreakLine { continue } preIsBreakLine = text == "" builder.WriteString(scanner.Text() + "\n") } if err := scanner.Err(); err != nil { fmt.Println(err) } return builder.String() }