Merge pull request #283 from colincross/uppercase_properties
Support unpacking capitalized property names
This commit is contained in:
commit
d851df9ada
3 changed files with 123 additions and 0 deletions
|
@ -16,19 +16,33 @@ package proptools
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"reflect"
|
"reflect"
|
||||||
|
"strings"
|
||||||
"unicode"
|
"unicode"
|
||||||
"unicode/utf8"
|
"unicode/utf8"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// PropertyNameForField converts the name of a field in property struct to the property name that
|
||||||
|
// might appear in a Blueprints file. Since the property struct fields must always be exported
|
||||||
|
// to be accessed with reflection and the canonical Blueprints style is lowercased names, it
|
||||||
|
// lower cases the first rune in the field name unless the field name contains multiple runes none
|
||||||
|
// of which are lowercase, in which case it returns the field name as-is.
|
||||||
func PropertyNameForField(fieldName string) string {
|
func PropertyNameForField(fieldName string) string {
|
||||||
r, size := utf8.DecodeRuneInString(fieldName)
|
r, size := utf8.DecodeRuneInString(fieldName)
|
||||||
propertyName := string(unicode.ToLower(r))
|
propertyName := string(unicode.ToLower(r))
|
||||||
|
if size == len(fieldName) {
|
||||||
|
return propertyName
|
||||||
|
}
|
||||||
|
if strings.IndexFunc(fieldName[size:], unicode.IsLower) == -1 {
|
||||||
|
return fieldName
|
||||||
|
}
|
||||||
if len(fieldName) > size {
|
if len(fieldName) > size {
|
||||||
propertyName += fieldName[size:]
|
propertyName += fieldName[size:]
|
||||||
}
|
}
|
||||||
return propertyName
|
return propertyName
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// FieldNameForProperty converts the name of a property that might appear in a Blueprints file to
|
||||||
|
// the name of a field in property struct by uppercasing the first rune.
|
||||||
func FieldNameForProperty(propertyName string) string {
|
func FieldNameForProperty(propertyName string) string {
|
||||||
r, size := utf8.DecodeRuneInString(propertyName)
|
r, size := utf8.DecodeRuneInString(propertyName)
|
||||||
fieldName := string(unicode.ToUpper(r))
|
fieldName := string(unicode.ToUpper(r))
|
||||||
|
|
94
proptools/proptools_test.go
Normal file
94
proptools/proptools_test.go
Normal file
|
@ -0,0 +1,94 @@
|
||||||
|
// Copyright 2020 Google Inc. All rights reserved.
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
package proptools
|
||||||
|
|
||||||
|
import "testing"
|
||||||
|
|
||||||
|
func TestPropertyNameForField(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
input string
|
||||||
|
want string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "short",
|
||||||
|
input: "S",
|
||||||
|
want: "s",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "long",
|
||||||
|
input: "String",
|
||||||
|
want: "string",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "uppercase",
|
||||||
|
input: "STRING",
|
||||||
|
want: "STRING",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "mixed",
|
||||||
|
input: "StRiNg",
|
||||||
|
want: "stRiNg",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
if got := PropertyNameForField(tt.input); got != tt.want {
|
||||||
|
t.Errorf("PropertyNameForField(%v) = %v, want %v", tt.input, got, tt.want)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFieldNameForProperty(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
input string
|
||||||
|
want string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "short lowercase",
|
||||||
|
input: "s",
|
||||||
|
want: "S",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "short uppercase",
|
||||||
|
input: "S",
|
||||||
|
want: "S",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "long lowercase",
|
||||||
|
input: "string",
|
||||||
|
want: "String",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "long uppercase",
|
||||||
|
input: "STRING",
|
||||||
|
want: "STRING",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "mixed",
|
||||||
|
input: "StRiNg",
|
||||||
|
want: "StRiNg",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
if got := FieldNameForProperty(tt.input); got != tt.want {
|
||||||
|
t.Errorf("FieldNameForProperty(%v) = %v, want %v", tt.input, got, tt.want)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
|
@ -508,6 +508,21 @@ var validUnpackTestCases = []struct {
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
// Captitalized property
|
||||||
|
{
|
||||||
|
input: `
|
||||||
|
m {
|
||||||
|
CAPITALIZED: "foo",
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
output: []interface{}{
|
||||||
|
&struct {
|
||||||
|
CAPITALIZED string
|
||||||
|
}{
|
||||||
|
CAPITALIZED: "foo",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestUnpackProperties(t *testing.T) {
|
func TestUnpackProperties(t *testing.T) {
|
||||||
|
|
Loading…
Reference in a new issue