mirror of
https://github.com/zeromicro/go-zero.git
synced 2025-01-24 01:30:25 +08:00
157 lines
3.4 KiB
Go
157 lines
3.4 KiB
Go
package format
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"strings"
|
|
)
|
|
|
|
const (
|
|
flagGo = "GO"
|
|
flagZero = "ZERO"
|
|
|
|
unknown style = iota
|
|
title
|
|
lower
|
|
upper
|
|
)
|
|
|
|
// ErrNamingFormat defines an error for unknown format
|
|
var ErrNamingFormat = errors.New("unsupported format")
|
|
|
|
type (
|
|
styleFormat struct {
|
|
before string
|
|
through string
|
|
after string
|
|
goStyle style
|
|
zeroStyle style
|
|
}
|
|
|
|
style int
|
|
)
|
|
|
|
// FileNamingFormat is used to format the file name. You can define the format style
|
|
// through the go and zero formatting characters. For example, you can define the snake
|
|
// format as go_zero, and the camel case format as goZero. You can even specify the split
|
|
// character, such as go#Zero, theoretically any combination can be used, but the prerequisite
|
|
// must meet the naming conventions of each operating system file name.
|
|
// Note: Formatting is based on snake or camel string
|
|
func FileNamingFormat(format, content string) (string, error) {
|
|
upperFormat := strings.ToUpper(format)
|
|
indexGo := strings.Index(upperFormat, flagGo)
|
|
indexZero := strings.Index(upperFormat, flagZero)
|
|
if indexGo < 0 || indexZero < 0 || indexGo > indexZero {
|
|
return "", ErrNamingFormat
|
|
}
|
|
var (
|
|
before, through, after string
|
|
flagGo, flagZero string
|
|
goStyle, zeroStyle style
|
|
err error
|
|
)
|
|
before = format[:indexGo]
|
|
flagGo = format[indexGo : indexGo+2]
|
|
through = format[indexGo+2 : indexZero]
|
|
flagZero = format[indexZero : indexZero+4]
|
|
after = format[indexZero+4:]
|
|
|
|
goStyle, err = getStyle(flagGo)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
zeroStyle, err = getStyle(flagZero)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
var formatStyle styleFormat
|
|
formatStyle.goStyle = goStyle
|
|
formatStyle.zeroStyle = zeroStyle
|
|
formatStyle.before = before
|
|
formatStyle.through = through
|
|
formatStyle.after = after
|
|
return doFormat(formatStyle, content)
|
|
}
|
|
|
|
func doFormat(f styleFormat, content string) (string, error) {
|
|
splits, err := split(content)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
var join []string
|
|
for index, split := range splits {
|
|
if index == 0 {
|
|
join = append(join, transferTo(split, f.goStyle))
|
|
continue
|
|
}
|
|
join = append(join, transferTo(split, f.zeroStyle))
|
|
}
|
|
joined := strings.Join(join, f.through)
|
|
return f.before + joined + f.after, nil
|
|
}
|
|
|
|
func transferTo(in string, style style) string {
|
|
switch style {
|
|
case upper:
|
|
return strings.ToUpper(in)
|
|
case lower:
|
|
return strings.ToLower(in)
|
|
case title:
|
|
return strings.Title(in)
|
|
default:
|
|
return in
|
|
}
|
|
}
|
|
|
|
func split(content string) ([]string, error) {
|
|
var (
|
|
list []string
|
|
reader = strings.NewReader(content)
|
|
buffer = bytes.NewBuffer(nil)
|
|
)
|
|
for {
|
|
r, _, err := reader.ReadRune()
|
|
if err != nil {
|
|
if err == io.EOF {
|
|
if buffer.Len() > 0 {
|
|
list = append(list, buffer.String())
|
|
}
|
|
return list, nil
|
|
}
|
|
return nil, err
|
|
}
|
|
if r == '_' {
|
|
if buffer.Len() > 0 {
|
|
list = append(list, buffer.String())
|
|
}
|
|
buffer.Reset()
|
|
continue
|
|
}
|
|
|
|
if r >= 'A' && r <= 'Z' {
|
|
if buffer.Len() > 0 {
|
|
list = append(list, buffer.String())
|
|
}
|
|
buffer.Reset()
|
|
}
|
|
buffer.WriteRune(r)
|
|
}
|
|
}
|
|
|
|
func getStyle(flag string) (style, error) {
|
|
compare := strings.ToLower(flag)
|
|
switch flag {
|
|
case strings.ToLower(compare):
|
|
return lower, nil
|
|
case strings.ToUpper(compare):
|
|
return upper, nil
|
|
case strings.Title(compare):
|
|
return title, nil
|
|
default:
|
|
return unknown, fmt.Errorf("unexpected format: %s", flag)
|
|
}
|
|
}
|