123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180 |
- package format
- import (
- "bufio"
- "errors"
- "fmt"
- "go/format"
- "go/scanner"
- "io/ioutil"
- "os"
- "path/filepath"
- "regexp"
- "strconv"
- "strings"
- "github.com/tal-tech/go-zero/core/errorx"
- "github.com/tal-tech/go-zero/tools/goctl/api/parser"
- "github.com/tal-tech/go-zero/tools/goctl/api/util"
- "github.com/urfave/cli"
- )
- var (
- reg = regexp.MustCompile("type (?P<name>.*)[\\s]+{")
- )
- func GoFormatApi(c *cli.Context) error {
- useStdin := c.Bool("stdin")
- var be errorx.BatchError
- if useStdin {
- if err := ApiFormatByStdin(); err != nil {
- be.Add(err)
- }
- } else {
- dir := c.String("dir")
- 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") {
- if err := ApiFormatByPath(path); err != nil {
- be.Add(util.WrapErr(err, fi.Name()))
- }
- }
- return nil
- })
- be.Add(err)
- }
- if be.NotNil() {
- scanner.PrintError(os.Stderr, be.Err())
- os.Exit(1)
- }
- return be.Err()
- }
- func ApiFormatByStdin() error {
- data, err := ioutil.ReadAll(os.Stdin)
- if err != nil {
- return err
- }
- result, err := apiFormat(string(data))
- if err != nil {
- return err
- }
- _, err = fmt.Print(result)
- if err != nil {
- return err
- }
- return nil
- }
- func ApiFormatByPath(apiFilePath string) error {
- data, err := ioutil.ReadFile(apiFilePath)
- if err != nil {
- return err
- }
- result, err := apiFormat(string(data))
- if err != nil {
- return err
- }
- if err := ioutil.WriteFile(apiFilePath, []byte(result), os.ModePerm); err != nil {
- return err
- }
- return nil
- }
- func apiFormat(data string) (string, error) {
- r := reg.ReplaceAllStringFunc(data, func(m string) string {
- parts := reg.FindStringSubmatch(m)
- if len(parts) < 2 {
- return m
- }
- if !strings.Contains(m, "struct") {
- return "type " + parts[1] + " struct {"
- }
- return m
- })
- apiStruct, err := parser.ParseApi(r)
- if err != nil {
- return "", err
- }
- info := strings.TrimSpace(apiStruct.Info)
- if len(apiStruct.Service) == 0 {
- return data, nil
- }
- fs, err := format.Source([]byte(strings.TrimSpace(apiStruct.StructBody)))
- if err != nil {
- str := err.Error()
- lineNumber := strings.Index(str, ":")
- if lineNumber > 0 {
- ln, err := strconv.ParseInt(str[:lineNumber], 10, 64)
- if err != nil {
- return "", err
- }
- pn := 0
- if len(info) > 0 {
- pn = countRune(info, '\n') + 1
- }
- number := int(ln) + pn + 1
- return "", errors.New(fmt.Sprintf("line: %d, %s", number, str[lineNumber+1:]))
- }
- return "", err
- }
- var result string
- if len(strings.TrimSpace(info)) > 0 {
- result += strings.TrimSpace(info) + "\n\n"
- }
- if len(strings.TrimSpace(apiStruct.Imports)) > 0 {
- result += strings.TrimSpace(apiStruct.Imports) + "\n\n"
- }
- if len(strings.TrimSpace(string(fs))) > 0 {
- result += strings.TrimSpace(string(fs)) + "\n\n"
- }
- if len(strings.TrimSpace(apiStruct.Service)) > 0 {
- result += formatService(apiStruct.Service) + "\n\n"
- }
- return strings.TrimSpace(result), nil
- }
- func formatService(str string) string {
- var builder strings.Builder
- scanner := bufio.NewScanner(strings.NewReader(str))
- var tapCount = 0
- for scanner.Scan() {
- line := strings.TrimSpace(scanner.Text())
- if line == ")" || line == "}" {
- tapCount -= 1
- }
- util.WriteIndent(&builder, tapCount)
- builder.WriteString(line + "\n")
- if strings.HasSuffix(line, "(") || strings.HasSuffix(line, "{") {
- tapCount += 1
- }
- }
- return strings.TrimSpace(builder.String())
- }
- func countRune(s string, r rune) int {
- count := 0
- for _, c := range s {
- if c == r {
- count++
- }
- }
- return count
- }
|