123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783 |
- /**
- * Package validator
- *
- * MISC:
- * - anonymous structs - they don't have names so expect the Struct name within StructErrors to be blank
- *
- */
-
- package validator
-
- import (
- "bytes"
- "errors"
- "fmt"
- "reflect"
- "strings"
- "sync"
- "time"
- )
-
- const (
- utf8HexComma = "0x2C"
- utf8Pipe = "0x7C"
- tagSeparator = ","
- orSeparator = "|"
- tagKeySeparator = "="
- structOnlyTag = "structonly"
- noStructLevelTag = "nostructlevel"
- omitempty = "omitempty"
- skipValidationTag = "-"
- diveTag = "dive"
- existsTag = "exists"
- fieldErrMsg = "Key: '%s' Error:Field validation for '%s' failed on the '%s' tag"
- arrayIndexFieldName = "%s" + leftBracket + "%d" + rightBracket
- mapIndexFieldName = "%s" + leftBracket + "%v" + rightBracket
- invalidValidation = "Invalid validation tag on field %s"
- undefinedValidation = "Undefined validation function on field %s"
- validatorNotInitialized = "Validator instance not initialized"
- fieldNameRequired = "Field Name Required"
- tagRequired = "Tag Required"
- )
-
- var (
- timeType = reflect.TypeOf(time.Time{})
- timePtrType = reflect.TypeOf(&time.Time{})
- defaultCField = new(cField)
- )
-
- // StructLevel contains all of the information and helper methods
- // for reporting errors during struct level validation
- type StructLevel struct {
- TopStruct reflect.Value
- CurrentStruct reflect.Value
- errPrefix string
- nsPrefix string
- errs ValidationErrors
- v *Validate
- }
-
- // ReportValidationErrors accepts the key relative to the top level struct and validatin errors.
- // Example: had a triple nested struct User, ContactInfo, Country and ran errs := validate.Struct(country)
- // from within a User struct level validation would call this method like so:
- // ReportValidationErrors("ContactInfo.", errs)
- // NOTE: relativeKey can contain both the Field Relative and Custom name relative paths
- // i.e. ReportValidationErrors("ContactInfo.|cInfo", errs) where cInfo represents say the JSON name of
- // the relative path; this will be split into 2 variables in the next valiator version.
- func (sl *StructLevel) ReportValidationErrors(relativeKey string, errs ValidationErrors) {
- for _, e := range errs {
-
- idx := strings.Index(relativeKey, "|")
- var rel string
- var cRel string
-
- if idx != -1 {
- rel = relativeKey[:idx]
- cRel = relativeKey[idx+1:]
- } else {
- rel = relativeKey
- }
-
- key := sl.errPrefix + rel + e.Field
-
- e.FieldNamespace = key
- e.NameNamespace = sl.nsPrefix + cRel + e.Name
-
- sl.errs[key] = e
- }
- }
-
- // ReportError reports an error just by passing the field and tag information
- // NOTE: tag can be an existing validation tag or just something you make up
- // and precess on the flip side it's up to you.
- func (sl *StructLevel) ReportError(field reflect.Value, fieldName string, customName string, tag string) {
-
- field, kind := sl.v.ExtractType(field)
-
- if fieldName == blank {
- panic(fieldNameRequired)
- }
-
- if customName == blank {
- customName = fieldName
- }
-
- if tag == blank {
- panic(tagRequired)
- }
-
- ns := sl.errPrefix + fieldName
-
- switch kind {
- case reflect.Invalid:
- sl.errs[ns] = &FieldError{
- FieldNamespace: ns,
- NameNamespace: sl.nsPrefix + customName,
- Name: customName,
- Field: fieldName,
- Tag: tag,
- ActualTag: tag,
- Param: blank,
- Kind: kind,
- }
- default:
- sl.errs[ns] = &FieldError{
- FieldNamespace: ns,
- NameNamespace: sl.nsPrefix + customName,
- Name: customName,
- Field: fieldName,
- Tag: tag,
- ActualTag: tag,
- Param: blank,
- Value: field.Interface(),
- Kind: kind,
- Type: field.Type(),
- }
- }
- }
-
- // Validate contains the validator settings passed in using the Config struct
- type Validate struct {
- tagName string
- fieldNameTag string
- validationFuncs map[string]Func
- structLevelFuncs map[reflect.Type]StructLevelFunc
- customTypeFuncs map[reflect.Type]CustomTypeFunc
- aliasValidators map[string]string
- hasCustomFuncs bool
- hasAliasValidators bool
- hasStructLevelFuncs bool
- tagCache *tagCache
- structCache *structCache
- errsPool *sync.Pool
- }
-
- func (v *Validate) initCheck() {
- if v == nil {
- panic(validatorNotInitialized)
- }
- }
-
- // Config contains the options that a Validator instance will use.
- // It is passed to the New() function
- type Config struct {
- TagName string
- FieldNameTag string
- }
-
- // CustomTypeFunc allows for overriding or adding custom field type handler functions
- // field = field value of the type to return a value to be validated
- // example Valuer from sql drive see https://golang.org/src/database/sql/driver/types.go?s=1210:1293#L29
- type CustomTypeFunc func(field reflect.Value) interface{}
-
- // Func accepts all values needed for file and cross field validation
- // v = validator instance, needed but some built in functions for it's custom types
- // topStruct = top level struct when validating by struct otherwise nil
- // currentStruct = current level struct when validating by struct otherwise optional comparison value
- // field = field value for validation
- // param = parameter used in validation i.e. gt=0 param would be 0
- type Func func(v *Validate, topStruct reflect.Value, currentStruct reflect.Value, field reflect.Value, fieldtype reflect.Type, fieldKind reflect.Kind, param string) bool
-
- // StructLevelFunc accepts all values needed for struct level validation
- type StructLevelFunc func(v *Validate, structLevel *StructLevel)
-
- // ValidationErrors is a type of map[string]*FieldError
- // it exists to allow for multiple errors to be passed from this library
- // and yet still subscribe to the error interface
- type ValidationErrors map[string]*FieldError
-
- // Error is intended for use in development + debugging and not intended to be a production error message.
- // It allows ValidationErrors to subscribe to the Error interface.
- // All information to create an error message specific to your application is contained within
- // the FieldError found within the ValidationErrors map
- func (ve ValidationErrors) Error() string {
-
- buff := bytes.NewBufferString(blank)
-
- for key, err := range ve {
- buff.WriteString(fmt.Sprintf(fieldErrMsg, key, err.Field, err.Tag))
- buff.WriteString("\n")
- }
-
- return strings.TrimSpace(buff.String())
- }
-
- // FieldError contains a single field's validation error along
- // with other properties that may be needed for error message creation
- type FieldError struct {
- FieldNamespace string
- NameNamespace string
- Field string
- Name string
- Tag string
- ActualTag string
- Kind reflect.Kind
- Type reflect.Type
- Param string
- Value interface{}
- }
-
- // New creates a new Validate instance for use.
- func New(config *Config) *Validate {
-
- tc := new(tagCache)
- tc.m.Store(make(map[string]*cTag))
-
- sc := new(structCache)
- sc.m.Store(make(map[reflect.Type]*cStruct))
-
- v := &Validate{
- tagName: config.TagName,
- fieldNameTag: config.FieldNameTag,
- tagCache: tc,
- structCache: sc,
- errsPool: &sync.Pool{New: func() interface{} {
- return ValidationErrors{}
- }}}
-
- if len(v.aliasValidators) == 0 {
- // must copy alias validators for separate validations to be used in each validator instance
- v.aliasValidators = map[string]string{}
- for k, val := range bakedInAliasValidators {
- v.RegisterAliasValidation(k, val)
- }
- }
-
- if len(v.validationFuncs) == 0 {
- // must copy validators for separate validations to be used in each instance
- v.validationFuncs = map[string]Func{}
- for k, val := range bakedInValidators {
- v.RegisterValidation(k, val)
- }
- }
-
- return v
- }
-
- // RegisterStructValidation registers a StructLevelFunc against a number of types
- // NOTE: this method is not thread-safe it is intended that these all be registered prior to any validation
- func (v *Validate) RegisterStructValidation(fn StructLevelFunc, types ...interface{}) {
- v.initCheck()
-
- if v.structLevelFuncs == nil {
- v.structLevelFuncs = map[reflect.Type]StructLevelFunc{}
- }
-
- for _, t := range types {
- v.structLevelFuncs[reflect.TypeOf(t)] = fn
- }
-
- v.hasStructLevelFuncs = true
- }
-
- // RegisterValidation adds a validation Func to a Validate's map of validators denoted by the key
- // NOTE: if the key already exists, the previous validation function will be replaced.
- // NOTE: this method is not thread-safe it is intended that these all be registered prior to any validation
- func (v *Validate) RegisterValidation(key string, fn Func) error {
- v.initCheck()
-
- if key == blank {
- return errors.New("Function Key cannot be empty")
- }
-
- if fn == nil {
- return errors.New("Function cannot be empty")
- }
-
- _, ok := restrictedTags[key]
-
- if ok || strings.ContainsAny(key, restrictedTagChars) {
- panic(fmt.Sprintf(restrictedTagErr, key))
- }
-
- v.validationFuncs[key] = fn
-
- return nil
- }
-
- // RegisterCustomTypeFunc registers a CustomTypeFunc against a number of types
- // NOTE: this method is not thread-safe it is intended that these all be registered prior to any validation
- func (v *Validate) RegisterCustomTypeFunc(fn CustomTypeFunc, types ...interface{}) {
- v.initCheck()
-
- if v.customTypeFuncs == nil {
- v.customTypeFuncs = map[reflect.Type]CustomTypeFunc{}
- }
-
- for _, t := range types {
- v.customTypeFuncs[reflect.TypeOf(t)] = fn
- }
-
- v.hasCustomFuncs = true
- }
-
- // RegisterAliasValidation registers a mapping of a single validationstag that
- // defines a common or complex set of validation(s) to simplify adding validation
- // to structs. NOTE: when returning an error the tag returned in FieldError will be
- // the alias tag unless the dive tag is part of the alias; everything after the
- // dive tag is not reported as the alias tag. Also the ActualTag in the before case
- // will be the actual tag within the alias that failed.
- // NOTE: this method is not thread-safe it is intended that these all be registered prior to any validation
- func (v *Validate) RegisterAliasValidation(alias, tags string) {
- v.initCheck()
-
- _, ok := restrictedTags[alias]
-
- if ok || strings.ContainsAny(alias, restrictedTagChars) {
- panic(fmt.Sprintf(restrictedAliasErr, alias))
- }
-
- v.aliasValidators[alias] = tags
- v.hasAliasValidators = true
- }
-
- // Field validates a single field using tag style validation and returns nil or ValidationErrors as type error.
- // You will need to assert the error if it's not nil i.e. err.(validator.ValidationErrors) to access the map of errors.
- // NOTE: it returns ValidationErrors instead of a single FieldError because this can also
- // validate Array, Slice and maps fields which may contain more than one error
- func (v *Validate) Field(field interface{}, tag string) error {
- v.initCheck()
-
- if len(tag) == 0 || tag == skipValidationTag {
- return nil
- }
-
- errs := v.errsPool.Get().(ValidationErrors)
- fieldVal := reflect.ValueOf(field)
-
- ctag, ok := v.tagCache.Get(tag)
- if !ok {
- v.tagCache.lock.Lock()
- defer v.tagCache.lock.Unlock()
-
- // could have been multiple trying to access, but once first is done this ensures tag
- // isn't parsed again.
- ctag, ok = v.tagCache.Get(tag)
- if !ok {
- ctag, _ = v.parseFieldTagsRecursive(tag, blank, blank, false)
- v.tagCache.Set(tag, ctag)
- }
- }
-
- v.traverseField(fieldVal, fieldVal, fieldVal, blank, blank, errs, false, false, nil, nil, defaultCField, ctag)
-
- if len(errs) == 0 {
- v.errsPool.Put(errs)
- return nil
- }
-
- return errs
- }
-
- // FieldWithValue validates a single field, against another fields value using tag style validation and returns nil or ValidationErrors.
- // You will need to assert the error if it's not nil i.e. err.(validator.ValidationErrors) to access the map of errors.
- // NOTE: it returns ValidationErrors instead of a single FieldError because this can also
- // validate Array, Slice and maps fields which may contain more than one error
- func (v *Validate) FieldWithValue(val interface{}, field interface{}, tag string) error {
- v.initCheck()
-
- if len(tag) == 0 || tag == skipValidationTag {
- return nil
- }
-
- errs := v.errsPool.Get().(ValidationErrors)
- topVal := reflect.ValueOf(val)
-
- ctag, ok := v.tagCache.Get(tag)
- if !ok {
- v.tagCache.lock.Lock()
- defer v.tagCache.lock.Unlock()
-
- // could have been multiple trying to access, but once first is done this ensures tag
- // isn't parsed again.
- ctag, ok = v.tagCache.Get(tag)
- if !ok {
- ctag, _ = v.parseFieldTagsRecursive(tag, blank, blank, false)
- v.tagCache.Set(tag, ctag)
- }
- }
-
- v.traverseField(topVal, topVal, reflect.ValueOf(field), blank, blank, errs, false, false, nil, nil, defaultCField, ctag)
-
- if len(errs) == 0 {
- v.errsPool.Put(errs)
- return nil
- }
-
- return errs
- }
-
- // StructPartial validates the fields passed in only, ignoring all others.
- // Fields may be provided in a namespaced fashion relative to the struct provided
- // i.e. NestedStruct.Field or NestedArrayField[0].Struct.Name and returns nil or ValidationErrors as error
- // You will need to assert the error if it's not nil i.e. err.(validator.ValidationErrors) to access the map of errors.
- func (v *Validate) StructPartial(current interface{}, fields ...string) error {
- v.initCheck()
-
- sv, _ := v.ExtractType(reflect.ValueOf(current))
- name := sv.Type().Name()
- m := map[string]struct{}{}
-
- if fields != nil {
- for _, k := range fields {
-
- flds := strings.Split(k, namespaceSeparator)
- if len(flds) > 0 {
-
- key := name + namespaceSeparator
- for _, s := range flds {
-
- idx := strings.Index(s, leftBracket)
-
- if idx != -1 {
- for idx != -1 {
- key += s[:idx]
- m[key] = struct{}{}
-
- idx2 := strings.Index(s, rightBracket)
- idx2++
- key += s[idx:idx2]
- m[key] = struct{}{}
- s = s[idx2:]
- idx = strings.Index(s, leftBracket)
- }
- } else {
-
- key += s
- m[key] = struct{}{}
- }
-
- key += namespaceSeparator
- }
- }
- }
- }
-
- errs := v.errsPool.Get().(ValidationErrors)
-
- v.ensureValidStruct(sv, sv, sv, blank, blank, errs, true, len(m) != 0, false, m, false)
-
- if len(errs) == 0 {
- v.errsPool.Put(errs)
- return nil
- }
-
- return errs
- }
-
- // StructExcept validates all fields except the ones passed in.
- // Fields may be provided in a namespaced fashion relative to the struct provided
- // i.e. NestedStruct.Field or NestedArrayField[0].Struct.Name and returns nil or ValidationErrors as error
- // You will need to assert the error if it's not nil i.e. err.(validator.ValidationErrors) to access the map of errors.
- func (v *Validate) StructExcept(current interface{}, fields ...string) error {
- v.initCheck()
-
- sv, _ := v.ExtractType(reflect.ValueOf(current))
- name := sv.Type().Name()
- m := map[string]struct{}{}
-
- for _, key := range fields {
- m[name+namespaceSeparator+key] = struct{}{}
- }
-
- errs := v.errsPool.Get().(ValidationErrors)
-
- v.ensureValidStruct(sv, sv, sv, blank, blank, errs, true, len(m) != 0, true, m, false)
-
- if len(errs) == 0 {
- v.errsPool.Put(errs)
- return nil
- }
-
- return errs
- }
-
- // Struct validates a structs exposed fields, and automatically validates nested structs, unless otherwise specified.
- // it returns nil or ValidationErrors as error.
- // You will need to assert the error if it's not nil i.e. err.(validator.ValidationErrors) to access the map of errors.
- func (v *Validate) Struct(current interface{}) error {
- v.initCheck()
-
- errs := v.errsPool.Get().(ValidationErrors)
- sv := reflect.ValueOf(current)
-
- v.ensureValidStruct(sv, sv, sv, blank, blank, errs, true, false, false, nil, false)
-
- if len(errs) == 0 {
- v.errsPool.Put(errs)
- return nil
- }
-
- return errs
- }
-
- func (v *Validate) ensureValidStruct(topStruct reflect.Value, currentStruct reflect.Value, current reflect.Value, errPrefix string, nsPrefix string, errs ValidationErrors, useStructName bool, partial bool, exclude bool, includeExclude map[string]struct{}, isStructOnly bool) {
-
- if current.Kind() == reflect.Ptr && !current.IsNil() {
- current = current.Elem()
- }
-
- if current.Kind() != reflect.Struct && current.Kind() != reflect.Interface {
- panic("value passed for validation is not a struct")
- }
-
- v.tranverseStruct(topStruct, currentStruct, current, errPrefix, nsPrefix, errs, useStructName, partial, exclude, includeExclude, nil, nil)
- }
-
- // tranverseStruct traverses a structs fields and then passes them to be validated by traverseField
- func (v *Validate) tranverseStruct(topStruct reflect.Value, currentStruct reflect.Value, current reflect.Value, errPrefix string, nsPrefix string, errs ValidationErrors, useStructName bool, partial bool, exclude bool, includeExclude map[string]struct{}, cs *cStruct, ct *cTag) {
-
- var ok bool
- first := len(nsPrefix) == 0
- typ := current.Type()
-
- cs, ok = v.structCache.Get(typ)
- if !ok {
- cs = v.extractStructCache(current, typ.Name())
- }
-
- if useStructName {
- errPrefix += cs.Name + namespaceSeparator
-
- if len(v.fieldNameTag) != 0 {
- nsPrefix += cs.Name + namespaceSeparator
- }
- }
-
- // structonly tag present don't tranverseFields
- // but must still check and run below struct level validation
- // if present
- if first || ct == nil || ct.typeof != typeStructOnly {
-
- for _, f := range cs.fields {
-
- if partial {
-
- _, ok = includeExclude[errPrefix+f.Name]
-
- if (ok && exclude) || (!ok && !exclude) {
- continue
- }
- }
-
- v.traverseField(topStruct, currentStruct, current.Field(f.Idx), errPrefix, nsPrefix, errs, partial, exclude, includeExclude, cs, f, f.cTags)
- }
- }
-
- // check if any struct level validations, after all field validations already checked.
- if cs.fn != nil {
- cs.fn(v, &StructLevel{v: v, TopStruct: topStruct, CurrentStruct: current, errPrefix: errPrefix, nsPrefix: nsPrefix, errs: errs})
- }
- }
-
- // traverseField validates any field, be it a struct or single field, ensures it's validity and passes it along to be validated via it's tag options
- func (v *Validate) traverseField(topStruct reflect.Value, currentStruct reflect.Value, current reflect.Value, errPrefix string, nsPrefix string, errs ValidationErrors, partial bool, exclude bool, includeExclude map[string]struct{}, cs *cStruct, cf *cField, ct *cTag) {
-
- current, kind, nullable := v.extractTypeInternal(current, false)
- var typ reflect.Type
-
- switch kind {
- case reflect.Ptr, reflect.Interface, reflect.Invalid:
-
- if ct == nil {
- return
- }
-
- if ct.typeof == typeOmitEmpty {
- return
- }
-
- if ct.hasTag {
-
- ns := errPrefix + cf.Name
-
- if kind == reflect.Invalid {
- errs[ns] = &FieldError{
- FieldNamespace: ns,
- NameNamespace: nsPrefix + cf.AltName,
- Name: cf.AltName,
- Field: cf.Name,
- Tag: ct.aliasTag,
- ActualTag: ct.tag,
- Param: ct.param,
- Kind: kind,
- }
- return
- }
-
- errs[ns] = &FieldError{
- FieldNamespace: ns,
- NameNamespace: nsPrefix + cf.AltName,
- Name: cf.AltName,
- Field: cf.Name,
- Tag: ct.aliasTag,
- ActualTag: ct.tag,
- Param: ct.param,
- Value: current.Interface(),
- Kind: kind,
- Type: current.Type(),
- }
-
- return
- }
-
- case reflect.Struct:
- typ = current.Type()
-
- if typ != timeType {
-
- if ct != nil {
- ct = ct.next
- }
-
- if ct != nil && ct.typeof == typeNoStructLevel {
- return
- }
-
- v.tranverseStruct(topStruct, current, current, errPrefix+cf.Name+namespaceSeparator, nsPrefix+cf.AltName+namespaceSeparator, errs, false, partial, exclude, includeExclude, cs, ct)
- return
- }
- }
-
- if !ct.hasTag {
- return
- }
-
- typ = current.Type()
-
- OUTER:
- for {
- if ct == nil {
- return
- }
-
- switch ct.typeof {
-
- case typeExists:
- ct = ct.next
- continue
-
- case typeOmitEmpty:
-
- if !nullable && !HasValue(v, topStruct, currentStruct, current, typ, kind, blank) {
- return
- }
-
- ct = ct.next
- continue
-
- case typeDive:
-
- ct = ct.next
-
- // traverse slice or map here
- // or panic ;)
- switch kind {
- case reflect.Slice, reflect.Array:
-
- for i := 0; i < current.Len(); i++ {
- v.traverseField(topStruct, currentStruct, current.Index(i), errPrefix, nsPrefix, errs, partial, exclude, includeExclude, cs, &cField{Name: fmt.Sprintf(arrayIndexFieldName, cf.Name, i), AltName: fmt.Sprintf(arrayIndexFieldName, cf.AltName, i)}, ct)
- }
-
- case reflect.Map:
- for _, key := range current.MapKeys() {
- v.traverseField(topStruct, currentStruct, current.MapIndex(key), errPrefix, nsPrefix, errs, partial, exclude, includeExclude, cs, &cField{Name: fmt.Sprintf(mapIndexFieldName, cf.Name, key.Interface()), AltName: fmt.Sprintf(mapIndexFieldName, cf.AltName, key.Interface())}, ct)
- }
-
- default:
- // throw error, if not a slice or map then should not have gotten here
- // bad dive tag
- panic("dive error! can't dive on a non slice or map")
- }
-
- return
-
- case typeOr:
-
- errTag := blank
-
- for {
-
- if ct.fn(v, topStruct, currentStruct, current, typ, kind, ct.param) {
-
- // drain rest of the 'or' values, then continue or leave
- for {
-
- ct = ct.next
-
- if ct == nil {
- return
- }
-
- if ct.typeof != typeOr {
- continue OUTER
- }
- }
- }
-
- errTag += orSeparator + ct.tag
-
- if ct.next == nil {
- // if we get here, no valid 'or' value and no more tags
-
- ns := errPrefix + cf.Name
-
- if ct.hasAlias {
- errs[ns] = &FieldError{
- FieldNamespace: ns,
- NameNamespace: nsPrefix + cf.AltName,
- Name: cf.AltName,
- Field: cf.Name,
- Tag: ct.aliasTag,
- ActualTag: ct.actualAliasTag,
- Value: current.Interface(),
- Type: typ,
- Kind: kind,
- }
- } else {
- errs[errPrefix+cf.Name] = &FieldError{
- FieldNamespace: ns,
- NameNamespace: nsPrefix + cf.AltName,
- Name: cf.AltName,
- Field: cf.Name,
- Tag: errTag[1:],
- ActualTag: errTag[1:],
- Value: current.Interface(),
- Type: typ,
- Kind: kind,
- }
- }
-
- return
- }
-
- ct = ct.next
- }
-
- default:
- if !ct.fn(v, topStruct, currentStruct, current, typ, kind, ct.param) {
-
- ns := errPrefix + cf.Name
-
- errs[ns] = &FieldError{
- FieldNamespace: ns,
- NameNamespace: nsPrefix + cf.AltName,
- Name: cf.AltName,
- Field: cf.Name,
- Tag: ct.aliasTag,
- ActualTag: ct.tag,
- Value: current.Interface(),
- Param: ct.param,
- Type: typ,
- Kind: kind,
- }
-
- return
-
- }
-
- ct = ct.next
- }
- }
- }
|