Skip to content

Commit

Permalink
chore: Add ToLowerSnakeCase, use it for Protobuf field names
Browse files Browse the repository at this point in the history
Signed-off-by: Jeff Thompson <[email protected]>
  • Loading branch information
jefft0 committed Oct 9, 2023
1 parent fa8eb77 commit fb783ec
Show file tree
Hide file tree
Showing 2 changed files with 336 additions and 1 deletion.
2 changes: 1 addition & 1 deletion tm2/pkg/amino/genproto/genproto.go
Original file line number Diff line number Diff line change
Expand Up @@ -206,7 +206,7 @@ func (p3c *P3Context) GenerateProto3MessagePartial(p3doc *P3Doc, rt reflect.Type
p3Field := P3Field{
Repeated: fp3IsRepeated,
Type: fp3,
Name: field.Name,
Name: ToLowerSnakeCase(field.Name),
Number: field.FieldOptions.BinFieldNum,
}
p3msg.Fields = append(p3msg.Fields, p3Field)
Expand Down
335 changes: 335 additions & 0 deletions tm2/pkg/amino/genproto/snakecase.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,335 @@
// Copyright 2020-2023 Buf Technologies, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

// This file has ToLowerSnakeCase which the Protobuf linter uses to make
// the expected spelling of a message field name. It is a copy of:
// https://github.com/bufbuild/buf/blob/main/private/pkg/stringutil/stringutil.go

package genproto

import (
"sort"
"strings"
"unicode"
)

// TrimLines splits the output into individual lines and trims the spaces from each line.
//
// This also trims the start and end spaces from the original output.
func TrimLines(output string) string {
return strings.TrimSpace(strings.Join(SplitTrimLines(output), "\n"))
}

// SplitTrimLines splits the output into individual lines and trims the spaces from each line.
func SplitTrimLines(output string) []string {
// this should work for windows as well as \r will be trimmed
split := strings.Split(output, "\n")
lines := make([]string, len(split))
for i, line := range split {
lines[i] = strings.TrimSpace(line)
}
return lines
}

// SplitTrimLinesNoEmpty splits the output into individual lines and trims the spaces from each line.
//
// This removes any empty lines.
func SplitTrimLinesNoEmpty(output string) []string {
// this should work for windows as well as \r will be trimmed
split := strings.Split(output, "\n")
lines := make([]string, 0, len(split))
for _, line := range split {
line = strings.TrimSpace(line)
if line != "" {
lines = append(lines, line)
}
}
return lines
}

// MapToSortedSlice transforms m to a sorted slice.
func MapToSortedSlice(m map[string]struct{}) []string {
s := MapToSlice(m)
sort.Strings(s)
return s
}

// MapToSlice transforms m to a slice.
func MapToSlice(m map[string]struct{}) []string {
s := make([]string, 0, len(m))
for e := range m {
s = append(s, e)
}
return s
}

// SliceToMap transforms s to a map.
func SliceToMap(s []string) map[string]struct{} {
m := make(map[string]struct{}, len(s))
for _, e := range s {
m[e] = struct{}{}
}
return m
}

// SliceToUniqueSortedSlice returns a sorted copy of s with no duplicates.
func SliceToUniqueSortedSlice(s []string) []string {
return MapToSortedSlice(SliceToMap(s))
}

// SliceToUniqueSortedSliceFilterEmptyStrings returns a sorted copy of s with no duplicates and no empty strings.
//
// Strings with only spaces are considered empty.
func SliceToUniqueSortedSliceFilterEmptyStrings(s []string) []string {
m := SliceToMap(s)
for key := range m {
if strings.TrimSpace(key) == "" {
delete(m, key)
}
}
return MapToSortedSlice(m)
}

// SliceToChunks splits s into chunks of the given chunk size.
//
// If s is nil or empty, returns empty.
// If chunkSize is <=0, returns [][]string{s}.
func SliceToChunks(s []string, chunkSize int) [][]string {
var chunks [][]string
if len(s) == 0 {
return chunks
}
if chunkSize <= 0 {
return [][]string{s}
}
c := make([]string, len(s))
copy(c, s)
// https://github.com/golang/go/wiki/SliceTricks#batching-with-minimal-allocation
for chunkSize < len(c) {
c, chunks = c[chunkSize:], append(chunks, c[0:chunkSize:chunkSize])
}
return append(chunks, c)
}

// SliceElementsEqual returns true if the two slices have equal elements.
//
// Nil and empty slices are treated as equals.
func SliceElementsEqual(one []string, two []string) bool {
if len(one) != len(two) {
return false
}
for i, elem := range one {
if two[i] != elem {
return false
}
}
return true
}

// SliceElementsContained returns true if superset contains subset.
//
// Nil and empty slices are treated as equals.
func SliceElementsContained(superset []string, subset []string) bool {
m := SliceToMap(superset)
for _, elem := range subset {
if _, ok := m[elem]; !ok {
return false
}
}
return true
}

// JoinSliceQuoted joins the slice with quotes.
func JoinSliceQuoted(s []string, sep string) string {
if len(s) == 0 {
return ""
}
return `"` + strings.Join(s, `"`+sep+`"`) + `"`
}

// SliceToString prints the slice as [e1,e2].
func SliceToString(s []string) string {
if len(s) == 0 {
return ""
}
return "[" + strings.Join(s, ",") + "]"
}

// SliceToHumanString prints the slice as "e1, e2, and e3".
func SliceToHumanString(s []string) string {
switch len(s) {
case 0:
return ""
case 1:
return s[0]
case 2:
return s[0] + " and " + s[1]
default:
return strings.Join(s[:len(s)-1], ", ") + ", and " + s[len(s)-1]
}
}

// SliceToHumanStringQuoted prints the slice as `"e1", "e2", and "e3"`.
func SliceToHumanStringQuoted(s []string) string {
switch len(s) {
case 0:
return ""
case 1:
return `"` + s[0] + `"`
case 2:
return `"` + s[0] + `" and "` + s[1] + `"`
default:
return `"` + strings.Join(s[:len(s)-1], `", "`) + `", and "` + s[len(s)-1] + `"`
}
}

// SliceToHumanStringOr prints the slice as "e1, e2, or e3".
func SliceToHumanStringOr(s []string) string {
switch len(s) {
case 0:
return ""
case 1:
return s[0]
case 2:
return s[0] + " or " + s[1]
default:
return strings.Join(s[:len(s)-1], ", ") + ", or " + s[len(s)-1]
}
}

// SliceToHumanStringOrQuoted prints the slice as `"e1", "e2", or "e3"`.
func SliceToHumanStringOrQuoted(s []string) string {
switch len(s) {
case 0:
return ""
case 1:
return `"` + s[0] + `"`
case 2:
return `"` + s[0] + `" or "` + s[1] + `"`
default:
return `"` + strings.Join(s[:len(s)-1], `", "`) + `", or "` + s[len(s)-1] + `"`
}
}

// SnakeCaseOption is an option for snake_case conversions.
type SnakeCaseOption func(*snakeCaseOptions)

// SnakeCaseWithNewWordOnDigits is a SnakeCaseOption that signifies
// to split on digits, ie foo_bar_1 instead of foo_bar1.
func SnakeCaseWithNewWordOnDigits() SnakeCaseOption {
return func(snakeCaseOptions *snakeCaseOptions) {
snakeCaseOptions.newWordOnDigits = true
}
}

// ToLowerSnakeCase transforms s to lower_snake_case.
func ToLowerSnakeCase(s string, options ...SnakeCaseOption) string {
return strings.ToLower(toSnakeCase(s, options...))
}

// ToUpperSnakeCase transforms s to UPPER_SNAKE_CASE.
func ToUpperSnakeCase(s string, options ...SnakeCaseOption) string {
return strings.ToUpper(toSnakeCase(s, options...))
}

// ToPascalCase converts s to PascalCase.
//
// Splits on '-', '_', ' ', '\t', '\n', '\r'.
// Uppercase letters will stay uppercase,
func ToPascalCase(s string) string {
output := ""
var previous rune
for i, c := range strings.TrimSpace(s) {
if !isDelimiter(c) {
if i == 0 || isDelimiter(previous) || unicode.IsUpper(c) {
output += string(unicode.ToUpper(c))
} else {
output += string(unicode.ToLower(c))
}
}
previous = c
}
return output
}

// IsAlphanumeric returns true for [0-9a-zA-Z].
func IsAlphanumeric(r rune) bool {
return IsNumeric(r) || IsAlpha(r)
}

// IsAlpha returns true for [a-zA-Z].
func IsAlpha(r rune) bool {
return IsLowerAlpha(r) || IsUpperAlpha(r)
}

// IsLowerAlpha returns true for [a-z].
func IsLowerAlpha(r rune) bool {
return 'a' <= r && r <= 'z'
}

// IsUpperAlpha returns true for [A-Z].
func IsUpperAlpha(r rune) bool {
return 'A' <= r && r <= 'Z'
}

// IsNumeric returns true for [0-9].
func IsNumeric(r rune) bool {
return '0' <= r && r <= '9'
}

// IsLowerAlphanumeric returns true for [0-9a-z].
func IsLowerAlphanumeric(r rune) bool {
return IsNumeric(r) || IsLowerAlpha(r)
}

func toSnakeCase(s string, options ...SnakeCaseOption) string {
snakeCaseOptions := &snakeCaseOptions{}
for _, option := range options {
option(snakeCaseOptions)
}
output := ""
s = strings.TrimFunc(s, isDelimiter)
for i, c := range s {
if isDelimiter(c) {
c = '_'
}
if i == 0 {
output += string(c)
} else if isSnakeCaseNewWord(c, snakeCaseOptions.newWordOnDigits) &&
output[len(output)-1] != '_' &&
((i < len(s)-1 && !isSnakeCaseNewWord(rune(s[i+1]), true) && !isDelimiter(rune(s[i+1]))) ||
(snakeCaseOptions.newWordOnDigits && unicode.IsDigit(c)) ||
(unicode.IsLower(rune(s[i-1])))) {
output += "_" + string(c)
} else if !(isDelimiter(c) && output[len(output)-1] == '_') {
output += string(c)
}
}
return output
}

func isSnakeCaseNewWord(r rune, newWordOnDigits bool) bool {
if newWordOnDigits {
return unicode.IsUpper(r) || unicode.IsDigit(r)
}
return unicode.IsUpper(r)
}

func isDelimiter(r rune) bool {
return r == '.' || r == '-' || r == '_' || r == ' ' || r == '\t' || r == '\n' || r == '\r'
}

type snakeCaseOptions struct {
newWordOnDigits bool
}

0 comments on commit fb783ec

Please sign in to comment.