diff --git a/.github/workflows/go.yml b/.github/workflows/go.yml index e54a9df..03e7456 100644 --- a/.github/workflows/go.yml +++ b/.github/workflows/go.yml @@ -13,12 +13,12 @@ jobs: runs-on: ubuntu-latest steps: - name: Set up Go 1.x - uses: actions/setup-go@v2 + uses: actions/setup-go@v5 with: - go-version: ^1.13 + go-version: '^1.13' - name: Check out code into the Go module directory - uses: actions/checkout@v2 + uses: actions/checkout@v4 - name: Build run: go build -v . @@ -27,4 +27,4 @@ jobs: run: go test -v . - name: Lint - uses: golangci/golangci-lint-action@v3.2.0 + uses: golangci/golangci-lint-action@v6.1.0 diff --git a/.golangci.yml b/.golangci.yml index c0057a4..9e8e5fa 100644 --- a/.golangci.yml +++ b/.golangci.yml @@ -29,17 +29,14 @@ issues: text: "does not use range value in test Run" linters-settings: - nolintlint: - # Disable to ensure that nolint directives don't have a leading space. Default is true. - allow-leading-space: false exhaustive: default-signifies-exhaustive: true govet: - check-shadowing: true + enable-all: true + disable: + - fieldalignment cyclop: max-complexity: 25 - maligned: - suggest-new: true dupl: threshold: 150 goconst: @@ -52,6 +49,7 @@ linters-settings: linters: enable-all: true disable: + - depguard - nlreturn - gci - gochecknoinits @@ -61,16 +59,11 @@ linters: - testpackage - wsl - gomnd - - goerr113 # most of the errors here are meant for humans + - err113 - goheader - - exhaustivestruct - thelper - gocyclo # replaced by cyclop since it also calculates the package complexity - - maligned # replaced by govet 'fieldalignment' - - interfacer # deprecated - - scopelint # deprecated, replaced by exportloopref - wrapcheck # a little bit too much for k6, maybe after https://github.com/tomarrell/wrapcheck/issues/2 is fixed - - golint # this linter is deprecated - varnamelen - ireturn - tagliatelle @@ -80,5 +73,4 @@ linters: - grouper - decorder - nonamedreturns - - nosnakecase fast: false \ No newline at end of file diff --git a/example_test.go b/example_test.go index 2030854..f16790b 100644 --- a/example_test.go +++ b/example_test.go @@ -34,7 +34,12 @@ func ExampleGetField() { if err != nil { log.Fatal(err) } + fmt.Println(value) + + // output: + // first value + // third value } } @@ -62,6 +67,10 @@ func ExampleGetFieldKind() { log.Fatal(err) } fmt.Println(secondFieldKind) + + // output: + // string + // int } func ExampleGetFieldType() { @@ -88,6 +97,10 @@ func ExampleGetFieldType() { log.Fatal(err) } fmt.Println(secondFieldType) + + // output: + // string + // int } func ExampleGetFieldTag() { @@ -104,6 +117,10 @@ func ExampleGetFieldTag() { log.Fatal(err) } fmt.Println(tag) + + // output: + // first tag + // third tag } func ExampleHasField() { @@ -120,6 +137,10 @@ func ExampleHasField() { // has == false has, _ = reflections.HasField(s, "FourthField") fmt.Println(has) + + // output: + // true + // false } func ExampleFields() { @@ -136,6 +157,9 @@ func ExampleFields() { // []string{"FirstField", "SecondField", "ThirdField"} fields, _ = reflections.Fields(s) fmt.Println(fields) + + // output: + // [MyEmbeddedStruct FirstField SecondField ThirdField] } func ExampleItems() { @@ -151,6 +175,9 @@ func ExampleItems() { // field value map structItems, _ = reflections.Items(s) fmt.Println(structItems) + + // output: + // map[FirstField:first value MyEmbeddedStruct:{} SecondField:2 ThirdField:third value] } func ExampleItemsDeep() { @@ -170,6 +197,9 @@ func ExampleItemsDeep() { // anonymous embedded structs structItems, _ = reflections.ItemsDeep(s) fmt.Println(structItems) + + // output: + // map[EmbeddedField:embedded value FirstField:first value SecondField:2 ThirdField:third value] } func ExampleTags() { @@ -191,6 +221,9 @@ func ExampleTags() { // } structTags, _ = reflections.Tags(s, "matched") fmt.Println(structTags) + + // output: + // map[FirstField:first tag MyEmbeddedStruct: SecondField:second tag ThirdField:] } func ExampleSetField() { @@ -207,12 +240,11 @@ func ExampleSetField() { log.Fatal(err) } - // If you try to set a field's value using the wrong type, + // Note that if you try to set a field's value using the wrong type, // an error will be returned - err = reflections.SetField(&s, "FirstField", 123) // err != nil - if err != nil { - log.Fatal(err) - } + _ = reflections.SetField(&s, "FirstField", 123) // err != nil + + // output: } func ExampleGetFieldNameByTagValue() { diff --git a/go.mod b/go.mod index b9373be..fb7b66d 100644 --- a/go.mod +++ b/go.mod @@ -1,5 +1,11 @@ module github.com/oleiade/reflections -go 1.15 +go 1.22.6 -require github.com/stretchr/testify v1.6.1 +require github.com/stretchr/testify v1.9.0 + +require ( + github.com/davecgh/go-spew v1.1.1 // indirect + github.com/pmezard/go-difflib v1.0.0 // indirect + gopkg.in/yaml.v3 v3.0.1 // indirect +) diff --git a/go.sum b/go.sum index afe7890..60ce688 100644 --- a/go.sum +++ b/go.sum @@ -1,11 +1,10 @@ -github.com/davecgh/go-spew v1.1.0 h1:ZDRjVQ15GmhC3fiQ8ni8+OwkZQO4DARzQgrnXU1Liz8= -github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= +github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= -github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= -github.com/stretchr/testify v1.6.1 h1:hDPOHmpOpP40lSULcqw7IrRb/u7w6RpDC9399XyoNd0= -github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg= +github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= -gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo= -gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= +gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= diff --git a/reflections.go b/reflections.go index d3c457d..90c0ac2 100644 --- a/reflections.go +++ b/reflections.go @@ -112,7 +112,7 @@ func GetFieldNameByTagValue(obj interface{}, tagKey, tagValue string) (string, e objType := objValue.Type() fieldsCount := objType.NumField() - for i := 0; i < fieldsCount; i++ { + for i := range fieldsCount { structField := objType.Field(i) if structField.Tag.Get(tagKey) == tagValue { return structField.Name, nil @@ -142,7 +142,7 @@ func SetField(obj interface{}, name string, value interface{}) error { structFieldType := structFieldValue.Type() val := reflect.ValueOf(value) if !val.Type().AssignableTo(structFieldType) { - invalidTypeError := fmt.Errorf("provided value type not assignable to obj field type") + invalidTypeError := errors.New("provided value type not assignable to obj field type") return invalidTypeError } @@ -175,7 +175,7 @@ func Fields(obj interface{}) ([]string, error) { // FieldsDeep returns "flattened" fields. // -// Note that FieldsDeept treats fields from anonymous inner structs as normal fields. +// Note that FieldsDeep treats fields from anonymous inner structs as normal fields. func FieldsDeep(obj interface{}) ([]string, error) { return fields(obj, true) } @@ -190,7 +190,7 @@ func fields(obj interface{}, deep bool) ([]string, error) { fieldsCount := objType.NumField() var allFields []string - for i := 0; i < fieldsCount; i++ { + for i := range fieldsCount { field := objType.Field(i) if isExportableField(field) { if !deep || !field.Anonymous { @@ -233,7 +233,7 @@ func items(obj interface{}, deep bool) (map[string]interface{}, error) { allItems := make(map[string]interface{}) - for i := 0; i < fieldsCount; i++ { + for i := range fieldsCount { field := objType.Field(i) fieldValue := objValue.Field(i) @@ -281,7 +281,7 @@ func tags(obj interface{}, key string, deep bool) (map[string]string, error) { allTags := make(map[string]string) - for i := 0; i < fieldsCount; i++ { + for i := range fieldsCount { structField := objType.Field(i) if isExportableField(structField) { if !deep || !structField.Anonymous { diff --git a/reflections_test.go b/reflections_test.go index badf693..6cf8975 100644 --- a/reflections_test.go +++ b/reflections_test.go @@ -9,12 +9,13 @@ import ( "testing" "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" ) type TestStruct struct { - unexported uint64 Dummy string `test:"dummytag"` - Yummy int `test:"yummytag"` + unexported uint64 + Yummy int `test:"yummytag"` } func TestGetField_on_struct(t *testing.T) { @@ -25,8 +26,8 @@ func TestGetField_on_struct(t *testing.T) { } value, err := GetField(dummyStruct, "Dummy") - assert.NoError(t, err) - assert.Equal(t, value, "test") + require.NoError(t, err) + assert.Equal(t, "test", value) } func TestGetField_on_struct_pointer(t *testing.T) { @@ -37,8 +38,8 @@ func TestGetField_on_struct_pointer(t *testing.T) { } value, err := GetField(dummyStruct, "Dummy") - assert.NoError(t, err) - assert.Equal(t, value, "test") + require.NoError(t, err) + assert.Equal(t, "test", value) } func TestGetField_on_non_struct(t *testing.T) { @@ -83,12 +84,12 @@ func TestGetFieldKind_on_struct(t *testing.T) { } kind, err := GetFieldKind(dummyStruct, "Dummy") - assert.NoError(t, err) - assert.Equal(t, kind, reflect.String) + require.NoError(t, err) + assert.Equal(t, reflect.String, kind) kind, err = GetFieldKind(dummyStruct, "Yummy") - assert.NoError(t, err) - assert.Equal(t, kind, reflect.Int) + require.NoError(t, err) + assert.Equal(t, reflect.Int, kind) } func TestGetFieldKind_on_struct_pointer(t *testing.T) { @@ -100,12 +101,12 @@ func TestGetFieldKind_on_struct_pointer(t *testing.T) { } kind, err := GetFieldKind(dummyStruct, "Dummy") - assert.NoError(t, err) - assert.Equal(t, kind, reflect.String) + require.NoError(t, err) + assert.Equal(t, reflect.String, kind) kind, err = GetFieldKind(dummyStruct, "Yummy") - assert.NoError(t, err) - assert.Equal(t, kind, reflect.Int) + require.NoError(t, err) + assert.Equal(t, reflect.Int, kind) } func TestGetFieldKind_on_non_struct(t *testing.T) { @@ -138,12 +139,12 @@ func TestGetFieldType_on_struct(t *testing.T) { } typeString, err := GetFieldType(dummyStruct, "Dummy") - assert.NoError(t, err) - assert.Equal(t, typeString, "string") + require.NoError(t, err) + assert.Equal(t, "string", typeString) typeString, err = GetFieldType(dummyStruct, "Yummy") - assert.NoError(t, err) - assert.Equal(t, typeString, "int") + require.NoError(t, err) + assert.Equal(t, "int", typeString) } func TestGetFieldType_on_struct_pointer(t *testing.T) { @@ -155,12 +156,12 @@ func TestGetFieldType_on_struct_pointer(t *testing.T) { } typeString, err := GetFieldType(dummyStruct, "Dummy") - assert.NoError(t, err) - assert.Equal(t, typeString, "string") + require.NoError(t, err) + assert.Equal(t, "string", typeString) typeString, err = GetFieldType(dummyStruct, "Yummy") - assert.NoError(t, err) - assert.Equal(t, typeString, "int") + require.NoError(t, err) + assert.Equal(t, "int", typeString) } func TestGetFieldType_on_non_struct(t *testing.T) { @@ -190,12 +191,12 @@ func TestGetFieldTag_on_struct(t *testing.T) { dummyStruct := TestStruct{} tag, err := GetFieldTag(dummyStruct, "Dummy", "test") - assert.NoError(t, err) - assert.Equal(t, tag, "dummytag") + require.NoError(t, err) + assert.Equal(t, "dummytag", tag) tag, err = GetFieldTag(dummyStruct, "Yummy", "test") - assert.NoError(t, err) - assert.Equal(t, tag, "yummytag") + require.NoError(t, err) + assert.Equal(t, "yummytag", tag) } func TestGetFieldTag_on_struct_pointer(t *testing.T) { @@ -204,12 +205,12 @@ func TestGetFieldTag_on_struct_pointer(t *testing.T) { dummyStruct := &TestStruct{} tag, err := GetFieldTag(dummyStruct, "Dummy", "test") - assert.NoError(t, err) - assert.Equal(t, tag, "dummytag") + require.NoError(t, err) + assert.Equal(t, "dummytag", tag) tag, err = GetFieldTag(dummyStruct, "Yummy", "test") - assert.NoError(t, err) - assert.Equal(t, tag, "yummytag") + require.NoError(t, err) + assert.Equal(t, "yummytag", tag) } func TestGetFieldTag_on_non_struct(t *testing.T) { @@ -250,8 +251,8 @@ func TestSetField_on_struct_with_valid_value_type(t *testing.T) { } err := SetField(&dummyStruct, "Dummy", "abc") - assert.NoError(t, err) - assert.Equal(t, dummyStruct.Dummy, "abc") + require.NoError(t, err) + assert.Equal(t, "abc", dummyStruct.Dummy) } func TestSetField_non_existing_field(t *testing.T) { @@ -295,8 +296,8 @@ func TestFields_on_struct(t *testing.T) { } fields, err := Fields(dummyStruct) - assert.NoError(t, err) - assert.Equal(t, fields, []string{"Dummy", "Yummy"}) + require.NoError(t, err) + assert.Equal(t, []string{"Dummy", "Yummy"}, fields) } func TestFields_on_struct_pointer(t *testing.T) { @@ -308,8 +309,8 @@ func TestFields_on_struct_pointer(t *testing.T) { } fields, err := Fields(dummyStruct) - assert.NoError(t, err) - assert.Equal(t, fields, []string{"Dummy", "Yummy"}) + require.NoError(t, err) + assert.Equal(t, []string{"Dummy", "Yummy"}, fields) } func TestFields_on_non_struct(t *testing.T) { @@ -331,8 +332,8 @@ func TestFields_with_non_exported_fields(t *testing.T) { } fields, err := Fields(dummyStruct) - assert.NoError(t, err) - assert.Equal(t, fields, []string{"Dummy", "Yummy"}) + require.NoError(t, err) + assert.Equal(t, []string{"Dummy", "Yummy"}, fields) } func TestHasField_on_struct_with_existing_field(t *testing.T) { @@ -344,7 +345,7 @@ func TestHasField_on_struct_with_existing_field(t *testing.T) { } has, err := HasField(dummyStruct, "Dummy") - assert.NoError(t, err) + require.NoError(t, err) assert.True(t, has) } @@ -357,7 +358,7 @@ func TestHasField_on_struct_pointer_with_existing_field(t *testing.T) { } has, err := HasField(dummyStruct, "Dummy") - assert.NoError(t, err) + require.NoError(t, err) assert.True(t, has) } @@ -370,7 +371,7 @@ func TestHasField_non_existing_field(t *testing.T) { } has, err := HasField(dummyStruct, "Test") - assert.NoError(t, err) + require.NoError(t, err) assert.False(t, has) } @@ -393,7 +394,7 @@ func TestHasField_unexported_field(t *testing.T) { } has, err := HasField(dummyStruct, "unexported") - assert.NoError(t, err) + require.NoError(t, err) assert.False(t, has) } @@ -406,11 +407,11 @@ func TestTags_on_struct(t *testing.T) { } tags, err := Tags(dummyStruct, "test") - assert.NoError(t, err) - assert.Equal(t, tags, map[string]string{ + require.NoError(t, err) + assert.Equal(t, map[string]string{ "Dummy": "dummytag", "Yummy": "yummytag", - }) + }, tags) } func TestTags_on_struct_pointer(t *testing.T) { @@ -422,11 +423,11 @@ func TestTags_on_struct_pointer(t *testing.T) { } tags, err := Tags(dummyStruct, "test") - assert.NoError(t, err) - assert.Equal(t, tags, map[string]string{ + require.NoError(t, err) + assert.Equal(t, map[string]string{ "Dummy": "dummytag", "Yummy": "yummytag", - }) + }, tags) } func TestTags_on_non_struct(t *testing.T) { @@ -447,11 +448,11 @@ func TestItems_on_struct(t *testing.T) { } tags, err := Items(dummyStruct) - assert.NoError(t, err) - assert.Equal(t, tags, map[string]interface{}{ + require.NoError(t, err) + assert.Equal(t, map[string]interface{}{ "Dummy": "test", "Yummy": 123, - }) + }, tags) } func TestItems_on_struct_pointer(t *testing.T) { @@ -463,11 +464,11 @@ func TestItems_on_struct_pointer(t *testing.T) { } tags, err := Items(dummyStruct) - assert.NoError(t, err) - assert.Equal(t, tags, map[string]interface{}{ + require.NoError(t, err) + assert.Equal(t, map[string]interface{}{ "Dummy": "test", "Yummy": 123, - }) + }, tags) } func TestItems_on_non_struct(t *testing.T) { @@ -502,15 +503,15 @@ func TestItems_deep(t *testing.T) { p.Number = 17 items, err := Items(p) - assert.NoError(t, err) + require.NoError(t, err) itemsDeep, err := ItemsDeep(p) - assert.NoError(t, err) + require.NoError(t, err) - assert.Equal(t, len(items), 2) - assert.Equal(t, len(itemsDeep), 3) - assert.Equal(t, itemsDeep["Name"], "John") - assert.Equal(t, itemsDeep["Street"], "Decumanus maximus") - assert.Equal(t, itemsDeep["Number"], 17) + assert.Len(t, items, 2) + assert.Len(t, itemsDeep, 3) + assert.Equal(t, "John", itemsDeep["Name"]) + assert.Equal(t, "Decumanus maximus", itemsDeep["Street"]) + assert.Equal(t, 17, itemsDeep["Number"]) } func TestGetFieldNameByTagValue(t *testing.T) { @@ -524,8 +525,8 @@ func TestGetFieldNameByTagValue(t *testing.T) { tagJSON := "dummytag" field, err := GetFieldNameByTagValue(dummyStruct, "test", tagJSON) - assert.NoError(t, err) - assert.Equal(t, field, "Dummy") + require.NoError(t, err) + assert.Equal(t, "Dummy", field) } func TestGetFieldNameByTagValue_on_non_existing_tag(t *testing.T) { @@ -539,17 +540,17 @@ func TestGetFieldNameByTagValue_on_non_existing_tag(t *testing.T) { // non existing tag value with an existing tag key tagJSON := "tag" _, errTagValue := GetFieldNameByTagValue(dummyStruct, "test", tagJSON) - assert.Error(t, errTagValue) + require.Error(t, errTagValue) // non existing tag key with an existing tag value tagJSON = "dummytag" _, errTagKey := GetFieldNameByTagValue(dummyStruct, "json", tagJSON) - assert.Error(t, errTagKey) + require.Error(t, errTagKey) // non existing tag key and value tagJSON = "tag" _, errTagKeyValue := GetFieldNameByTagValue(dummyStruct, "json", tagJSON) - assert.Error(t, errTagKeyValue) + require.Error(t, errTagKeyValue) } //nolint:unused @@ -575,15 +576,15 @@ func TestTags_deep(t *testing.T) { p.Number = 17 tags, err := Tags(p, "tag") - assert.NoError(t, err) + require.NoError(t, err) tagsDeep, err := TagsDeep(p, "tag") - assert.NoError(t, err) + require.NoError(t, err) - assert.Equal(t, len(tags), 2) - assert.Equal(t, len(tagsDeep), 3) - assert.Equal(t, tagsDeep["Name"], "bu") - assert.Equal(t, tagsDeep["Street"], "be") - assert.Equal(t, tagsDeep["Number"], "bi") + assert.Len(t, tags, 2) + assert.Len(t, tagsDeep, 3) + assert.Equal(t, "bu", tagsDeep["Name"]) + assert.Equal(t, "be", tagsDeep["Street"]) + assert.Equal(t, "bi", tagsDeep["Number"]) } //nolint:unused @@ -609,15 +610,15 @@ func TestFields_deep(t *testing.T) { p.Number = 17 fields, err := Fields(p) - assert.NoError(t, err) + require.NoError(t, err) fieldsDeep, err := FieldsDeep(p) - assert.NoError(t, err) + require.NoError(t, err) - assert.Equal(t, len(fields), 2) - assert.Equal(t, len(fieldsDeep), 3) - assert.Equal(t, fieldsDeep[0], "Name") - assert.Equal(t, fieldsDeep[1], "Street") - assert.Equal(t, fieldsDeep[2], "Number") + assert.Len(t, fields, 2) + assert.Len(t, fieldsDeep, 3) + assert.Equal(t, "Name", fieldsDeep[0]) + assert.Equal(t, "Street", fieldsDeep[1]) + assert.Equal(t, "Number", fieldsDeep[2]) } type SingleString string @@ -629,13 +630,15 @@ type Bar struct { } func TestAssignable(t *testing.T) { + t.Parallel() + var b Bar expected := []string{"a", "b", "c"} - assert.Nil(t, SetField(&b, "A", expected)) + require.NoError(t, SetField(&b, "A", expected)) assert.Equal(t, StringList(expected), b.A) err := SetField(&b, "A", []int{0, 1, 2}) - assert.NotNil(t, err) + require.Error(t, err) assert.Equal(t, "provided value type not assignable to obj field type", err.Error()) }