2020-07-29 17:11:41 +08:00
|
|
|
package tsgen
|
|
|
|
|
|
|
|
|
|
import (
|
2024-06-25 23:18:15 +08:00
|
|
|
"bytes"
|
2021-01-11 15:10:51 +08:00
|
|
|
"errors"
|
2020-07-29 17:11:41 +08:00
|
|
|
"fmt"
|
|
|
|
|
"io"
|
|
|
|
|
"strings"
|
|
|
|
|
|
2022-01-25 23:15:07 +08:00
|
|
|
"github.com/zeromicro/go-zero/tools/goctl/api/spec"
|
|
|
|
|
apiutil "github.com/zeromicro/go-zero/tools/goctl/api/util"
|
|
|
|
|
"github.com/zeromicro/go-zero/tools/goctl/util"
|
2020-07-29 17:11:41 +08:00
|
|
|
)
|
|
|
|
|
|
2022-09-24 22:28:25 +08:00
|
|
|
const (
|
|
|
|
|
formTagKey = "form"
|
|
|
|
|
pathTagKey = "path"
|
|
|
|
|
headerTagKey = "header"
|
|
|
|
|
)
|
|
|
|
|
|
2021-01-11 15:10:51 +08:00
|
|
|
func writeProperty(writer io.Writer, member spec.Member, indent int) error {
|
2020-07-29 17:11:41 +08:00
|
|
|
writeIndent(writer, indent)
|
2024-06-25 23:18:15 +08:00
|
|
|
ty, err := genTsType(member, indent)
|
2020-08-19 16:00:55 +08:00
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-29 17:11:41 +08:00
|
|
|
optionalTag := ""
|
2021-02-26 16:11:47 +08:00
|
|
|
if member.IsOptional() || member.IsOmitEmpty() {
|
2020-07-29 17:11:41 +08:00
|
|
|
optionalTag = "?"
|
|
|
|
|
}
|
|
|
|
|
name, err := member.GetPropertyName()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
2020-08-19 16:00:55 +08:00
|
|
|
|
2025-05-02 17:15:32 +08:00
|
|
|
if strings.Contains(name, "-") {
|
|
|
|
|
name = fmt.Sprintf("'%s'", name)
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-29 17:11:41 +08:00
|
|
|
comment := member.GetComment()
|
|
|
|
|
if len(comment) > 0 {
|
|
|
|
|
comment = strings.TrimPrefix(comment, "//")
|
|
|
|
|
comment = " // " + strings.TrimSpace(comment)
|
|
|
|
|
}
|
|
|
|
|
if len(member.Docs) > 0 {
|
2020-08-19 16:00:55 +08:00
|
|
|
fmt.Fprintf(writer, "%s\n", strings.Join(member.Docs, ""))
|
2024-06-25 23:18:15 +08:00
|
|
|
writeIndent(writer, indent)
|
2020-07-29 17:11:41 +08:00
|
|
|
}
|
|
|
|
|
_, err = fmt.Fprintf(writer, "%s%s: %s%s\n", name, optionalTag, ty, comment)
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func writeIndent(writer io.Writer, indent int) {
|
|
|
|
|
for i := 0; i < indent; i++ {
|
|
|
|
|
fmt.Fprint(writer, "\t")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-06-25 23:18:15 +08:00
|
|
|
func genTsType(m spec.Member, indent int) (ty string, err error) {
|
2024-07-28 23:40:25 +08:00
|
|
|
v, ok := m.Type.(spec.NestedStruct)
|
|
|
|
|
if ok {
|
2024-06-25 23:18:15 +08:00
|
|
|
writer := bytes.NewBuffer(nil)
|
|
|
|
|
_, err := fmt.Fprintf(writer, "{\n")
|
|
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := writeMembers(writer, v, false, indent+1); err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
writeIndent(writer, indent)
|
|
|
|
|
_, err = fmt.Fprintf(writer, "}")
|
|
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
return writer.String(), nil
|
|
|
|
|
}
|
|
|
|
|
|
2022-10-11 22:19:22 +08:00
|
|
|
ty, err = goTypeToTs(m.Type, false)
|
|
|
|
|
if enums := m.GetEnumOptions(); enums != nil {
|
|
|
|
|
if ty == "string" {
|
|
|
|
|
for i := range enums {
|
|
|
|
|
enums[i] = "'" + enums[i] + "'"
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
ty = strings.Join(enums, " | ")
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-11 15:10:51 +08:00
|
|
|
func goTypeToTs(tp spec.Type, fromPacket bool) (string, error) {
|
|
|
|
|
switch v := tp.(type) {
|
|
|
|
|
case spec.DefineStruct:
|
|
|
|
|
return addPrefix(tp, fromPacket), nil
|
|
|
|
|
case spec.PrimitiveType:
|
|
|
|
|
r, ok := primitiveType(tp.Name())
|
|
|
|
|
if !ok {
|
|
|
|
|
return "", errors.New("unsupported primitive type " + tp.Name())
|
2020-07-29 17:11:41 +08:00
|
|
|
}
|
2021-01-11 15:10:51 +08:00
|
|
|
|
|
|
|
|
return r, nil
|
|
|
|
|
case spec.MapType:
|
|
|
|
|
valueType, err := goTypeToTs(v.Value, fromPacket)
|
2020-07-29 17:11:41 +08:00
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
2021-01-11 15:10:51 +08:00
|
|
|
|
|
|
|
|
return fmt.Sprintf("{ [key: string]: %s }", valueType), nil
|
|
|
|
|
case spec.ArrayType:
|
|
|
|
|
if tp.Name() == "[]byte" {
|
|
|
|
|
return "Blob", nil
|
2020-07-29 17:11:41 +08:00
|
|
|
}
|
2021-01-11 15:10:51 +08:00
|
|
|
|
|
|
|
|
valueType, err := goTypeToTs(v.Value, fromPacket)
|
2020-07-29 17:11:41 +08:00
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
2021-01-11 15:10:51 +08:00
|
|
|
|
|
|
|
|
return fmt.Sprintf("Array<%s>", valueType), nil
|
|
|
|
|
case spec.InterfaceType:
|
|
|
|
|
return "any", nil
|
|
|
|
|
case spec.PointerType:
|
|
|
|
|
return goTypeToTs(v.Type, fromPacket)
|
2020-07-29 17:11:41 +08:00
|
|
|
}
|
2021-01-11 15:10:51 +08:00
|
|
|
|
|
|
|
|
return "", errors.New("unsupported type " + tp.Name())
|
2020-07-29 17:11:41 +08:00
|
|
|
}
|
|
|
|
|
|
2021-01-11 15:10:51 +08:00
|
|
|
func addPrefix(tp spec.Type, fromPacket bool) string {
|
|
|
|
|
if fromPacket {
|
|
|
|
|
return packagePrefix + util.Title(tp.Name())
|
2020-07-29 17:11:41 +08:00
|
|
|
}
|
2021-01-11 15:10:51 +08:00
|
|
|
return util.Title(tp.Name())
|
2020-07-29 17:11:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func primitiveType(tp string) (string, bool) {
|
|
|
|
|
switch tp {
|
|
|
|
|
case "string":
|
|
|
|
|
return "string", true
|
2023-05-12 10:38:05 +08:00
|
|
|
case "int", "int8", "int16", "int32", "int64", "uint", "uint8", "uint16", "uint32", "uint64":
|
2020-07-29 17:11:41 +08:00
|
|
|
return "number", true
|
|
|
|
|
case "float", "float32", "float64":
|
|
|
|
|
return "number", true
|
|
|
|
|
case "bool":
|
|
|
|
|
return "boolean", true
|
|
|
|
|
case "[]byte":
|
|
|
|
|
return "Blob", true
|
|
|
|
|
case "interface{}":
|
|
|
|
|
return "any", true
|
|
|
|
|
}
|
|
|
|
|
return "", false
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-11 15:10:51 +08:00
|
|
|
func writeType(writer io.Writer, tp spec.Type) error {
|
|
|
|
|
fmt.Fprintf(writer, "export interface %s {\n", util.Title(tp.Name()))
|
2024-06-25 23:18:15 +08:00
|
|
|
if err := writeMembers(writer, tp, false, 1); err != nil {
|
2020-07-29 17:11:41 +08:00
|
|
|
return err
|
|
|
|
|
}
|
2021-01-11 15:10:51 +08:00
|
|
|
|
2020-07-29 17:11:41 +08:00
|
|
|
fmt.Fprintf(writer, "}\n")
|
2021-01-11 15:10:51 +08:00
|
|
|
return genParamsTypesIfNeed(writer, tp)
|
2020-07-29 17:11:41 +08:00
|
|
|
}
|
|
|
|
|
|
2021-01-11 15:10:51 +08:00
|
|
|
func genParamsTypesIfNeed(writer io.Writer, tp spec.Type) error {
|
2025-10-07 03:27:46 +00:00
|
|
|
_, ok := tp.(spec.DefineStruct)
|
2021-01-11 15:10:51 +08:00
|
|
|
if !ok {
|
|
|
|
|
return errors.New("no members of type " + tp.Name())
|
|
|
|
|
}
|
|
|
|
|
|
2025-10-07 03:27:46 +00:00
|
|
|
// Check if there are actual non-body members (recursively through inline structs)
|
|
|
|
|
if !hasActualNonBodyMembers(tp) {
|
2020-07-29 17:11:41 +08:00
|
|
|
return nil
|
|
|
|
|
}
|
2022-09-24 22:28:25 +08:00
|
|
|
|
2021-01-11 15:10:51 +08:00
|
|
|
fmt.Fprintf(writer, "export interface %sParams {\n", util.Title(tp.Name()))
|
2022-09-24 22:28:25 +08:00
|
|
|
if err := writeTagMembers(writer, tp, formTagKey); err != nil {
|
2020-07-29 17:11:41 +08:00
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
fmt.Fprintf(writer, "}\n")
|
2022-09-24 22:28:25 +08:00
|
|
|
|
2025-10-07 03:27:46 +00:00
|
|
|
if hasActualTagMembers(tp, headerTagKey) {
|
2022-09-24 22:28:25 +08:00
|
|
|
fmt.Fprintf(writer, "export interface %sHeaders {\n", util.Title(tp.Name()))
|
|
|
|
|
if err := writeTagMembers(writer, tp, headerTagKey); err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
fmt.Fprintf(writer, "}\n")
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-29 17:11:41 +08:00
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2024-06-25 23:18:15 +08:00
|
|
|
func writeMembers(writer io.Writer, tp spec.Type, isParam bool, indent int) error {
|
2021-01-11 15:10:51 +08:00
|
|
|
definedType, ok := tp.(spec.DefineStruct)
|
|
|
|
|
if !ok {
|
2021-01-13 11:54:53 +08:00
|
|
|
pointType, ok := tp.(spec.PointerType)
|
|
|
|
|
if ok {
|
2024-06-25 23:18:15 +08:00
|
|
|
return writeMembers(writer, pointType.Type, isParam, indent)
|
2021-01-13 11:54:53 +08:00
|
|
|
}
|
2021-02-09 13:50:21 +08:00
|
|
|
|
|
|
|
|
return fmt.Errorf("type %s not supported", tp.Name())
|
2021-01-11 15:10:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
members := definedType.GetBodyMembers()
|
2020-07-29 17:11:41 +08:00
|
|
|
if isParam {
|
2021-01-11 15:10:51 +08:00
|
|
|
members = definedType.GetNonBodyMembers()
|
2020-07-29 17:11:41 +08:00
|
|
|
}
|
|
|
|
|
for _, member := range members {
|
|
|
|
|
if member.IsInline {
|
2024-06-25 23:18:15 +08:00
|
|
|
if err := writeMembers(writer, member.Type, isParam, indent); err != nil {
|
2020-07-29 17:11:41 +08:00
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
continue
|
|
|
|
|
}
|
2021-01-11 15:10:51 +08:00
|
|
|
|
2024-06-25 23:18:15 +08:00
|
|
|
if err := writeProperty(writer, member, indent); err != nil {
|
2021-01-11 15:10:51 +08:00
|
|
|
return apiutil.WrapErr(err, " type "+tp.Name())
|
2020-07-29 17:11:41 +08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
}
|
2022-09-24 22:28:25 +08:00
|
|
|
|
|
|
|
|
func writeTagMembers(writer io.Writer, tp spec.Type, tagKey string) error {
|
|
|
|
|
definedType, ok := tp.(spec.DefineStruct)
|
|
|
|
|
if !ok {
|
|
|
|
|
pointType, ok := tp.(spec.PointerType)
|
|
|
|
|
if ok {
|
|
|
|
|
return writeTagMembers(writer, pointType.Type, tagKey)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return fmt.Errorf("type %s not supported", tp.Name())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
members := definedType.GetTagMembers(tagKey)
|
|
|
|
|
for _, member := range members {
|
|
|
|
|
if member.IsInline {
|
|
|
|
|
if err := writeTagMembers(writer, member.Type, tagKey); err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := writeProperty(writer, member, 1); err != nil {
|
|
|
|
|
return apiutil.WrapErr(err, " type "+tp.Name())
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
}
|
2025-10-07 03:27:46 +00:00
|
|
|
|
|
|
|
|
// hasActualTagMembers checks if a type has actual members with the given tag,
|
|
|
|
|
// recursively checking inline/embedded structs
|
|
|
|
|
func hasActualTagMembers(tp spec.Type, tagKey string) bool {
|
|
|
|
|
definedType, ok := tp.(spec.DefineStruct)
|
|
|
|
|
if !ok {
|
|
|
|
|
pointType, ok := tp.(spec.PointerType)
|
|
|
|
|
if ok {
|
|
|
|
|
return hasActualTagMembers(pointType.Type, tagKey)
|
|
|
|
|
}
|
|
|
|
|
return false
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, m := range definedType.Members {
|
|
|
|
|
if m.IsInline {
|
|
|
|
|
// Recursively check inline members
|
|
|
|
|
if hasActualTagMembers(m.Type, tagKey) {
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// Check non-inline members for the tag
|
|
|
|
|
if m.IsTagMember(tagKey) {
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// hasActualBodyMembers checks if a type has actual body members (json tags),
|
|
|
|
|
// recursively checking inline/embedded structs
|
|
|
|
|
func hasActualBodyMembers(tp spec.Type) bool {
|
|
|
|
|
definedType, ok := tp.(spec.DefineStruct)
|
|
|
|
|
if !ok {
|
|
|
|
|
pointType, ok := tp.(spec.PointerType)
|
|
|
|
|
if ok {
|
|
|
|
|
return hasActualBodyMembers(pointType.Type)
|
|
|
|
|
}
|
|
|
|
|
return false
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, m := range definedType.Members {
|
|
|
|
|
if m.IsInline {
|
|
|
|
|
// Recursively check inline members
|
|
|
|
|
if hasActualBodyMembers(m.Type) {
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// Check non-inline members for json tag
|
|
|
|
|
if m.IsBodyMember() {
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// hasActualNonBodyMembers checks if a type has actual non-body members (form, path, header tags),
|
|
|
|
|
// recursively checking inline/embedded structs
|
|
|
|
|
func hasActualNonBodyMembers(tp spec.Type) bool {
|
|
|
|
|
definedType, ok := tp.(spec.DefineStruct)
|
|
|
|
|
if !ok {
|
|
|
|
|
pointType, ok := tp.(spec.PointerType)
|
|
|
|
|
if ok {
|
|
|
|
|
return hasActualNonBodyMembers(pointType.Type)
|
|
|
|
|
}
|
|
|
|
|
return false
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, m := range definedType.Members {
|
|
|
|
|
if m.IsInline {
|
|
|
|
|
// Recursively check inline members
|
|
|
|
|
if hasActualNonBodyMembers(m.Type) {
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// Check non-inline members for non-body tags
|
|
|
|
|
if !m.IsBodyMember() {
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false
|
|
|
|
|
}
|