Skip to content

Commit 0d4b817

Browse files
authored
Merge pull request #191 from OscarVanL/fix/188/validate-embedded-structs
fixes #188
2 parents 97f1366 + 1d3b06e commit 0d4b817

File tree

3 files changed

+106
-5
lines changed

3 files changed

+106
-5
lines changed

data_source.go

+3-5
Original file line numberDiff line numberDiff line change
@@ -297,7 +297,6 @@ func (d *StructData) parseRulesFromTag(v *Validation) {
297297
// preStrName - the parent field name.
298298
recursiveFunc = func(vv reflect.Value, vt reflect.Type, parentFName string, parentIsAnonymous bool) {
299299
for i := 0; i < vt.NumField(); i++ {
300-
fValue := removeValuePtr(vv).Field(i)
301300
fv := vt.Field(i)
302301
// skip don't exported field
303302
name := fv.Name
@@ -365,16 +364,15 @@ func (d *StructData) parseRulesFromTag(v *Validation) {
365364
ft := removeTypePtr(vt.Field(i).Type)
366365

367366
// collect rules from sub-struct and from arrays/slices elements
368-
if ft != timeType {
369-
if fValue.Type().Kind() == reflect.Ptr && fValue.IsNil() {
370-
continue
371-
}
367+
if ft != timeType && removeValuePtr(vv).IsValid() {
372368

373369
// feat: only collect sub-struct rule on current field has rule.
374370
if vRule == "" && gOpt.CheckSubOnParentMarked {
375371
continue
376372
}
377373

374+
fValue := removeValuePtr(vv).Field(i)
375+
378376
switch ft.Kind() {
379377
case reflect.Struct:
380378
recursiveFunc(fValue, ft, name, fv.Anonymous)

issues_test.go

+95
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
package validate_test
22

33
import (
4+
"encoding/json"
45
"fmt"
56
"testing"
67
"time"
@@ -297,6 +298,100 @@ func TestPtrFieldValidation(t *testing.T) {
297298
assert.False(t, valid.Validate())
298299
}
299300

301+
type Pet struct {
302+
Breed string `json:"breed" validate:"min_len:3"`
303+
Color string `json:"color" validate:"in:orange,black,brown"`
304+
}
305+
type Human struct {
306+
Age int `json:"age" validate:"gt:13"`
307+
Name string `json:"name" validate:"min_len:3"`
308+
}
309+
type Settings struct {
310+
*Pet `json:",omitempty"`
311+
*Human `json:",omitempty"`
312+
}
313+
314+
type Entity struct {
315+
ID string `json:"id" validate:"uuid4"`
316+
Kind string `json:"kind" validate:"in:pet,human"`
317+
Settings Settings `json:"settings"`
318+
}
319+
320+
func (Entity) ConfigValidation(v *validate.Validation) {
321+
v.WithScenes(validate.SValues{
322+
"required": []string{"ID", "Kind"},
323+
"pet": []string{"Settings.Pet"},
324+
"human": []string{"Settings.Human"},
325+
})
326+
327+
}
328+
329+
func TestPtrFieldEmbeddedValid(t *testing.T) {
330+
// Valid case :)
331+
input := []byte(`{
332+
"id": "59fcf270-8646-4250-b4ff-d50f6121bc9d",
333+
"kind": "pet",
334+
"settings": {
335+
"breed": "dog",
336+
"color": "brown"
337+
}
338+
}`)
339+
var entity Entity
340+
err := json.Unmarshal(input, &entity)
341+
fmt.Println(entity)
342+
assert.NoError(t, err)
343+
344+
validate.Config(func(opt *validate.GlobalOption) {
345+
opt.SkipOnEmpty = false
346+
opt.StopOnError = false
347+
})
348+
349+
vld := validate.Struct(&entity)
350+
vld.SkipOnEmpty = false
351+
err = vld.ValidateE("required")
352+
assert.Empty(t, err)
353+
vld.ResetResult()
354+
err = vld.ValidateE(entity.Kind)
355+
assert.Empty(t, err)
356+
357+
validate.ResetOption()
358+
}
359+
360+
func TestPtrFieldEmbeddedInvalid(t *testing.T) {
361+
// Invalid case
362+
input := []byte(`{
363+
"id": "59fcf270-8646-4250-b4ff-d50f6121bc9d",
364+
"kind": "pet",
365+
"settings": {
366+
}
367+
}`)
368+
var entity Entity
369+
err := json.Unmarshal(input, &entity)
370+
fmt.Println(entity)
371+
assert.NoError(t, err)
372+
373+
validate.Config(func(opt *validate.GlobalOption) {
374+
opt.SkipOnEmpty = false
375+
opt.StopOnError = false
376+
})
377+
vld := validate.Struct(&entity)
378+
err = vld.ValidateE("required")
379+
assert.Empty(t, err)
380+
vld.ResetResult()
381+
err = vld.ValidateE(entity.Kind)
382+
expected := validate.Errors{
383+
"breed": validate.MS{
384+
"min_len": "breed min length is 3",
385+
},
386+
"color": validate.MS{
387+
"in": "color value must be in the enum [orange black brown]",
388+
},
389+
}
390+
391+
assert.Equal(t, expected, err)
392+
validate.ResetOption()
393+
}
394+
300395
// ----- test case structs
301396

302397
type Org struct {

validation.go

+8
Original file line numberDiff line numberDiff line change
@@ -591,6 +591,14 @@ func (v *Validation) isNotNeedToCheck(field string) bool {
591591
return false
592592
}
593593

594+
fields := strings.Split(field, ".")
595+
for i := 0; i < len(fields); i++ {
596+
_, ok := v.sceneFields[strings.Join(fields[0:i], ".")]
597+
if ok {
598+
return false
599+
}
600+
}
601+
594602
_, ok := v.sceneFields[field]
595603
return !ok
596604
}

0 commit comments

Comments
 (0)