2020-07-29 17:11:41 +08:00
|
|
|
package format
|
|
|
|
|
|
|
|
import (
|
2020-11-09 17:41:07 +08:00
|
|
|
"bufio"
|
2020-07-29 17:11:41 +08:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
2021-01-04 18:59:48 +08:00
|
|
|
"go/format"
|
2020-10-28 22:37:59 +08:00
|
|
|
"go/scanner"
|
2022-03-31 00:20:51 +08:00
|
|
|
"io"
|
2020-07-29 17:11:41 +08:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
|
2022-02-09 17:22:52 +08:00
|
|
|
"github.com/urfave/cli"
|
2022-01-25 23:15:07 +08:00
|
|
|
"github.com/zeromicro/go-zero/core/errorx"
|
|
|
|
"github.com/zeromicro/go-zero/tools/goctl/api/parser"
|
|
|
|
"github.com/zeromicro/go-zero/tools/goctl/api/util"
|
|
|
|
"github.com/zeromicro/go-zero/tools/goctl/util/pathx"
|
2020-07-29 17:11:41 +08:00
|
|
|
)
|
|
|
|
|
2020-11-17 18:08:55 +08:00
|
|
|
const (
|
|
|
|
leftParenthesis = "("
|
|
|
|
rightParenthesis = ")"
|
|
|
|
leftBrace = "{"
|
|
|
|
rightBrace = "}"
|
|
|
|
)
|
|
|
|
|
2021-02-26 16:11:47 +08:00
|
|
|
// GoFormatApi format api file
|
2020-07-29 17:11:41 +08:00
|
|
|
func GoFormatApi(c *cli.Context) error {
|
2020-10-28 22:37:59 +08:00
|
|
|
useStdin := c.Bool("stdin")
|
2022-03-12 15:17:31 +08:00
|
|
|
skipCheckDeclare := c.Bool("declare")
|
2022-03-31 00:20:51 +08:00
|
|
|
dir := c.String("dir")
|
2020-07-29 17:11:41 +08:00
|
|
|
|
|
|
|
var be errorx.BatchError
|
2020-10-28 22:37:59 +08:00
|
|
|
if useStdin {
|
2022-03-31 00:20:51 +08:00
|
|
|
if err := apiFormatReader(os.Stdin, dir, skipCheckDeclare); err != nil {
|
2020-10-28 22:37:59 +08:00
|
|
|
be.Add(err)
|
2020-07-29 17:11:41 +08:00
|
|
|
}
|
2020-10-28 22:37:59 +08:00
|
|
|
} else {
|
|
|
|
if len(dir) == 0 {
|
|
|
|
return errors.New("missing -dir")
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err := os.Lstat(dir)
|
|
|
|
if err != nil {
|
|
|
|
return errors.New(dir + ": No such file or directory")
|
|
|
|
}
|
|
|
|
|
|
|
|
err = filepath.Walk(dir, func(path string, fi os.FileInfo, errBack error) (err error) {
|
|
|
|
if strings.HasSuffix(path, ".api") {
|
2022-03-12 15:17:31 +08:00
|
|
|
if err := ApiFormatByPath(path, skipCheckDeclare); err != nil {
|
2020-10-28 22:37:59 +08:00
|
|
|
be.Add(util.WrapErr(err, fi.Name()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
be.Add(err)
|
|
|
|
}
|
2021-01-13 16:37:33 +08:00
|
|
|
|
2020-07-29 17:11:41 +08:00
|
|
|
if be.NotNil() {
|
2020-10-28 22:37:59 +08:00
|
|
|
scanner.PrintError(os.Stderr, be.Err())
|
2020-07-29 17:11:41 +08:00
|
|
|
os.Exit(1)
|
|
|
|
}
|
2021-01-13 16:37:33 +08:00
|
|
|
|
2020-07-29 17:11:41 +08:00
|
|
|
return be.Err()
|
|
|
|
}
|
|
|
|
|
2022-03-31 00:20:51 +08:00
|
|
|
// apiFormatReader
|
|
|
|
// filename is needed when there are `import` literals.
|
|
|
|
func apiFormatReader(reader io.Reader, filename string, skipCheckDeclare bool) error {
|
|
|
|
data, err := ioutil.ReadAll(reader)
|
2020-10-28 22:37:59 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-03-31 00:20:51 +08:00
|
|
|
result, err := apiFormat(string(data), skipCheckDeclare, filename)
|
2020-11-17 18:08:55 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-07-29 17:11:41 +08:00
|
|
|
|
2020-10-28 22:37:59 +08:00
|
|
|
_, err = fmt.Print(result)
|
2021-01-13 16:37:33 +08:00
|
|
|
return err
|
2020-10-28 22:37:59 +08:00
|
|
|
}
|
|
|
|
|
2021-02-26 16:11:47 +08:00
|
|
|
// ApiFormatByPath format api from file path
|
2022-03-12 15:17:31 +08:00
|
|
|
func ApiFormatByPath(apiFilePath string, skipCheckDeclare bool) error {
|
2020-10-28 22:37:59 +08:00
|
|
|
data, err := ioutil.ReadFile(apiFilePath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-05-13 18:42:05 +08:00
|
|
|
abs, err := filepath.Abs(apiFilePath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-03-12 15:17:31 +08:00
|
|
|
result, err := apiFormat(string(data), skipCheckDeclare, abs)
|
2020-11-17 18:08:55 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-03-12 15:17:31 +08:00
|
|
|
_, err = parser.ParseContentWithParserSkipCheckTypeDeclaration(result, abs)
|
2021-01-13 16:37:33 +08:00
|
|
|
if err != nil {
|
2020-10-28 22:37:59 +08:00
|
|
|
return err
|
|
|
|
}
|
2021-01-13 16:37:33 +08:00
|
|
|
|
|
|
|
return ioutil.WriteFile(apiFilePath, []byte(result), os.ModePerm)
|
2020-10-28 22:37:59 +08:00
|
|
|
}
|
|
|
|
|
2022-03-12 15:17:31 +08:00
|
|
|
func apiFormat(data string, skipCheckDeclare bool, filename ...string) (string, error) {
|
|
|
|
var err error
|
|
|
|
if skipCheckDeclare {
|
|
|
|
_, err = parser.ParseContentWithParserSkipCheckTypeDeclaration(data, filename...)
|
|
|
|
} else {
|
|
|
|
_, err = parser.ParseContent(data, filename...)
|
|
|
|
}
|
2020-11-17 18:08:55 +08:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
2020-11-09 17:41:07 +08:00
|
|
|
var builder strings.Builder
|
2020-11-17 18:08:55 +08:00
|
|
|
s := bufio.NewScanner(strings.NewReader(data))
|
2021-04-15 19:49:17 +08:00
|
|
|
tapCount := 0
|
|
|
|
newLineCount := 0
|
2021-01-04 18:59:48 +08:00
|
|
|
var preLine string
|
2020-11-17 18:08:55 +08:00
|
|
|
for s.Scan() {
|
|
|
|
line := strings.TrimSpace(s.Text())
|
2021-01-04 18:59:48 +08:00
|
|
|
if len(line) == 0 {
|
|
|
|
if newLineCount > 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
newLineCount++
|
|
|
|
} else {
|
|
|
|
if preLine == rightBrace {
|
2022-01-03 21:54:53 +08:00
|
|
|
builder.WriteString(pathx.NL)
|
2021-01-04 18:59:48 +08:00
|
|
|
}
|
|
|
|
newLineCount = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
if tapCount == 0 {
|
|
|
|
format, err := formatGoTypeDef(line, s, &builder)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
if format {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-17 15:25:13 +08:00
|
|
|
noCommentLine := util.RemoveComment(line)
|
2020-11-17 18:08:55 +08:00
|
|
|
if noCommentLine == rightParenthesis || noCommentLine == rightBrace {
|
2021-02-26 16:11:47 +08:00
|
|
|
tapCount--
|
2020-11-09 17:41:07 +08:00
|
|
|
}
|
2020-11-17 18:08:55 +08:00
|
|
|
if tapCount < 0 {
|
2021-01-04 18:59:48 +08:00
|
|
|
line := strings.TrimSuffix(noCommentLine, rightBrace)
|
2020-11-17 18:08:55 +08:00
|
|
|
line = strings.TrimSpace(line)
|
|
|
|
if strings.HasSuffix(line, leftBrace) {
|
2021-02-26 16:11:47 +08:00
|
|
|
tapCount++
|
2020-11-17 18:08:55 +08:00
|
|
|
}
|
|
|
|
}
|
2020-11-09 17:41:07 +08:00
|
|
|
util.WriteIndent(&builder, tapCount)
|
2022-01-03 21:54:53 +08:00
|
|
|
builder.WriteString(line + pathx.NL)
|
2020-11-17 18:08:55 +08:00
|
|
|
if strings.HasSuffix(noCommentLine, leftParenthesis) || strings.HasSuffix(noCommentLine, leftBrace) {
|
2021-02-26 16:11:47 +08:00
|
|
|
tapCount++
|
2020-11-09 17:41:07 +08:00
|
|
|
}
|
2021-01-04 18:59:48 +08:00
|
|
|
preLine = line
|
2020-11-09 17:41:07 +08:00
|
|
|
}
|
2021-01-13 16:37:33 +08:00
|
|
|
|
2020-11-17 18:08:55 +08:00
|
|
|
return strings.TrimSpace(builder.String()), nil
|
2020-07-29 17:11:41 +08:00
|
|
|
}
|
2021-01-04 18:59:48 +08:00
|
|
|
|
|
|
|
func formatGoTypeDef(line string, scanner *bufio.Scanner, builder *strings.Builder) (bool, error) {
|
|
|
|
noCommentLine := util.RemoveComment(line)
|
|
|
|
tokenCount := 0
|
|
|
|
if strings.HasPrefix(noCommentLine, "type") && (strings.HasSuffix(noCommentLine, leftParenthesis) ||
|
|
|
|
strings.HasSuffix(noCommentLine, leftBrace)) {
|
|
|
|
var typeBuilder strings.Builder
|
2022-01-03 21:54:53 +08:00
|
|
|
typeBuilder.WriteString(mayInsertStructKeyword(line, &tokenCount) + pathx.NL)
|
2021-01-04 18:59:48 +08:00
|
|
|
for scanner.Scan() {
|
|
|
|
noCommentLine := util.RemoveComment(scanner.Text())
|
2022-01-03 21:54:53 +08:00
|
|
|
typeBuilder.WriteString(mayInsertStructKeyword(scanner.Text(), &tokenCount) + pathx.NL)
|
2021-01-04 18:59:48 +08:00
|
|
|
if noCommentLine == rightBrace || noCommentLine == rightParenthesis {
|
|
|
|
tokenCount--
|
|
|
|
}
|
|
|
|
if tokenCount == 0 {
|
|
|
|
ts, err := format.Source([]byte(typeBuilder.String()))
|
|
|
|
if err != nil {
|
|
|
|
return false, errors.New("error format \n" + typeBuilder.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
result := strings.ReplaceAll(string(ts), " struct ", " ")
|
|
|
|
result = strings.ReplaceAll(result, "type ()", "")
|
|
|
|
builder.WriteString(result)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true, nil
|
|
|
|
}
|
2021-01-13 16:37:33 +08:00
|
|
|
|
2021-01-04 18:59:48 +08:00
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func mayInsertStructKeyword(line string, token *int) string {
|
|
|
|
insertStruct := func() string {
|
|
|
|
if strings.Contains(line, " struct") {
|
|
|
|
return line
|
|
|
|
}
|
|
|
|
index := strings.Index(line, leftBrace)
|
|
|
|
return line[:index] + " struct " + line[index:]
|
|
|
|
}
|
|
|
|
|
|
|
|
noCommentLine := util.RemoveComment(line)
|
|
|
|
if strings.HasSuffix(noCommentLine, leftBrace) {
|
|
|
|
*token++
|
|
|
|
return insertStruct()
|
|
|
|
}
|
|
|
|
if strings.HasSuffix(noCommentLine, rightBrace) {
|
|
|
|
noCommentLine = strings.TrimSuffix(noCommentLine, rightBrace)
|
|
|
|
noCommentLine = util.RemoveComment(noCommentLine)
|
|
|
|
if strings.HasSuffix(noCommentLine, leftBrace) {
|
|
|
|
return insertStruct()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if strings.HasSuffix(noCommentLine, leftParenthesis) {
|
|
|
|
*token++
|
|
|
|
}
|
2021-01-11 15:10:51 +08:00
|
|
|
|
|
|
|
if strings.Contains(noCommentLine, "`") {
|
|
|
|
return util.UpperFirst(strings.TrimSpace(line))
|
|
|
|
}
|
2021-01-13 16:37:33 +08:00
|
|
|
|
2021-01-04 18:59:48 +08:00
|
|
|
return line
|
|
|
|
}
|