github.com/angenalZZZ/gofunc@v0.0.0-20210507121333-48ff1be3917b/g/field.go (about)

     1  package g
     2  
     3  import (
     4  	"errors"
     5  	"fmt"
     6  	"reflect"
     7  )
     8  
     9  var (
    10  	ErrNotExported = errors.New("field is not exported")
    11  	ErrNotSettable = errors.New("field is not settable")
    12  )
    13  
    14  // Field represents a single struct field that encapsulates high level
    15  // functions around the field.
    16  type Field struct {
    17  	value      reflect.Value
    18  	field      reflect.StructField
    19  	defaultTag string
    20  }
    21  
    22  // Tag returns the value associated with key in the tag string. If there is no
    23  // such key in the tag, Tag returns the empty string.
    24  func (f *Field) Tag(key string) string {
    25  	return f.field.Tag.Get(key)
    26  }
    27  
    28  // Value returns the underlying value of the field. It panics if the field
    29  // is not exported.
    30  func (f *Field) Value() interface{} {
    31  	return f.value.Interface()
    32  }
    33  
    34  // IsEmbedded returns true if the given field is an anonymous field (embedded)
    35  func (f *Field) IsEmbedded() bool {
    36  	return f.field.Anonymous
    37  }
    38  
    39  // IsExported returns true if the given field is exported.
    40  func (f *Field) IsExported() bool {
    41  	return f.field.PkgPath == ""
    42  }
    43  
    44  // IsZero returns true if the given field is not initialized (has a zero value).
    45  // It panics if the field is not exported.
    46  func (f *Field) IsZero() bool {
    47  	zero := reflect.Zero(f.value.Type()).Interface()
    48  	current := f.Value()
    49  
    50  	return reflect.DeepEqual(current, zero)
    51  }
    52  
    53  // Name returns the name of the given field
    54  func (f *Field) Name() string {
    55  	return f.field.Name
    56  }
    57  
    58  // Kind returns the fields kind, such as "string", "map", "bool", etc ..
    59  func (f *Field) Kind() reflect.Kind {
    60  	return f.value.Kind()
    61  }
    62  
    63  // Set sets the field to given value v. It returns an error if the field is not
    64  // settable (not addressable or not exported) or if the given value's type
    65  // doesn't match the fields type.
    66  func (f *Field) Set(val interface{}) error {
    67  	// we can't set unexported fields, so be sure this field is exported
    68  	if !f.IsExported() {
    69  		return ErrNotExported
    70  	}
    71  
    72  	// do we get here? not sure...
    73  	if !f.value.CanSet() {
    74  		return ErrNotSettable
    75  	}
    76  
    77  	given := reflect.ValueOf(val)
    78  
    79  	if f.value.Kind() != given.Kind() {
    80  		return fmt.Errorf("wrong kind. got: %s want: %s", given.Kind(), f.value.Kind())
    81  	}
    82  
    83  	f.value.Set(given)
    84  	return nil
    85  }
    86  
    87  // Zero sets the field to its zero value. It returns an error if the field is not
    88  // settable (not addressable or not exported).
    89  func (f *Field) Zero() error {
    90  	zero := reflect.Zero(f.value.Type()).Interface()
    91  	return f.Set(zero)
    92  }
    93  
    94  // Fields returns a slice of Fields. This is particular handy to get the fields
    95  // of a nested struct . A struct tag with the content of "-" ignores the
    96  // checking of that particular field. Example:
    97  //
    98  //   // Field is ignored by this package.
    99  //   Field *http.Request `struct:"-"`
   100  //
   101  // It panics if field is not exported or if field's kind is not struct
   102  func (f *Field) Fields() []*Field {
   103  	return getFields(f.value, f.defaultTag)
   104  }
   105  
   106  // Field returns the field from a nested struct. It panics if the nested struct
   107  // is not exported or if the field was not found.
   108  func (f *Field) Field(name string) *Field {
   109  	field, ok := f.FieldOk(name)
   110  	if !ok {
   111  		panic("field not found")
   112  	}
   113  
   114  	return field
   115  }
   116  
   117  // FieldOk returns the field from a nested struct. The boolean returns whether
   118  // the field was found (true) or not (false).
   119  func (f *Field) FieldOk(name string) (*Field, bool) {
   120  	value := &f.value
   121  	// value must be settable so we need to make sure it holds the address of the
   122  	// variable and not a copy, so we can pass the pointer to structVal instead of a
   123  	// copy (which is not assigned to any variable, hence not settable).
   124  	// see "https://blog.golang.org/laws-of-reflection#TOC_8."
   125  	if f.value.Kind() != reflect.Ptr {
   126  		a := f.value.Addr()
   127  		value = &a
   128  	}
   129  	v := structVal(value.Interface())
   130  	t := v.Type()
   131  
   132  	field, ok := t.FieldByName(name)
   133  	if !ok {
   134  		return nil, false
   135  	}
   136  
   137  	return &Field{
   138  		field: field,
   139  		value: v.FieldByName(name),
   140  	}, true
   141  }