github.com/ethereum/go-ethereum@v1.16.1/rlp/internal/rlpstruct/rlpstruct.go (about) 1 // Copyright 2022 The go-ethereum Authors 2 // This file is part of the go-ethereum library. 3 // 4 // The go-ethereum library is free software: you can redistribute it and/or modify 5 // it under the terms of the GNU Lesser General Public License as published by 6 // the Free Software Foundation, either version 3 of the License, or 7 // (at your option) any later version. 8 // 9 // The go-ethereum library is distributed in the hope that it will be useful, 10 // but WITHOUT ANY WARRANTY; without even the implied warranty of 11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 // GNU Lesser General Public License for more details. 13 // 14 // You should have received a copy of the GNU Lesser General Public License 15 // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. 16 17 // Package rlpstruct implements struct processing for RLP encoding/decoding. 18 // 19 // In particular, this package handles all rules around field filtering, 20 // struct tags and nil value determination. 21 package rlpstruct 22 23 import ( 24 "fmt" 25 "reflect" 26 "strings" 27 ) 28 29 // Field represents a struct field. 30 type Field struct { 31 Name string 32 Index int 33 Exported bool 34 Type Type 35 Tag string 36 } 37 38 // Type represents the attributes of a Go type. 39 type Type struct { 40 Name string 41 Kind reflect.Kind 42 IsEncoder bool // whether type implements rlp.Encoder 43 IsDecoder bool // whether type implements rlp.Decoder 44 Elem *Type // non-nil for Kind values of Ptr, Slice, Array 45 } 46 47 // DefaultNilValue determines whether a nil pointer to t encodes/decodes 48 // as an empty string or empty list. 49 func (t Type) DefaultNilValue() NilKind { 50 k := t.Kind 51 if isUint(k) || k == reflect.String || k == reflect.Bool || isByteArray(t) { 52 return NilKindString 53 } 54 return NilKindList 55 } 56 57 // NilKind is the RLP value encoded in place of nil pointers. 58 type NilKind uint8 59 60 const ( 61 NilKindString NilKind = 0x80 62 NilKindList NilKind = 0xC0 63 ) 64 65 // Tags represents struct tags. 66 type Tags struct { 67 // rlp:"nil" controls whether empty input results in a nil pointer. 68 // nilKind is the kind of empty value allowed for the field. 69 NilKind NilKind 70 NilOK bool 71 72 // rlp:"optional" allows for a field to be missing in the input list. 73 // If this is set, all subsequent fields must also be optional. 74 Optional bool 75 76 // rlp:"tail" controls whether this field swallows additional list elements. It can 77 // only be set for the last field, which must be of slice type. 78 Tail bool 79 80 // rlp:"-" ignores fields. 81 Ignored bool 82 } 83 84 // TagError is raised for invalid struct tags. 85 type TagError struct { 86 StructType string 87 88 // These are set by this package. 89 Field string 90 Tag string 91 Err string 92 } 93 94 func (e TagError) Error() string { 95 field := "field " + e.Field 96 if e.StructType != "" { 97 field = e.StructType + "." + e.Field 98 } 99 return fmt.Sprintf("rlp: invalid struct tag %q for %s (%s)", e.Tag, field, e.Err) 100 } 101 102 // ProcessFields filters the given struct fields, returning only fields 103 // that should be considered for encoding/decoding. 104 func ProcessFields(allFields []Field) ([]Field, []Tags, error) { 105 lastPublic := lastPublicField(allFields) 106 107 // Gather all exported fields and their tags. 108 var fields []Field 109 var tags []Tags 110 for _, field := range allFields { 111 if !field.Exported { 112 continue 113 } 114 ts, err := parseTag(field, lastPublic) 115 if err != nil { 116 return nil, nil, err 117 } 118 if ts.Ignored { 119 continue 120 } 121 fields = append(fields, field) 122 tags = append(tags, ts) 123 } 124 125 // Verify optional field consistency. If any optional field exists, 126 // all fields after it must also be optional. Note: optional + tail 127 // is supported. 128 var anyOptional bool 129 var firstOptionalName string 130 for i, ts := range tags { 131 name := fields[i].Name 132 if ts.Optional || ts.Tail { 133 if !anyOptional { 134 firstOptionalName = name 135 } 136 anyOptional = true 137 } else { 138 if anyOptional { 139 msg := fmt.Sprintf("must be optional because preceding field %q is optional", firstOptionalName) 140 return nil, nil, TagError{Field: name, Err: msg} 141 } 142 } 143 } 144 return fields, tags, nil 145 } 146 147 func parseTag(field Field, lastPublic int) (Tags, error) { 148 name := field.Name 149 tag := reflect.StructTag(field.Tag) 150 var ts Tags 151 for _, t := range strings.Split(tag.Get("rlp"), ",") { 152 switch t = strings.TrimSpace(t); t { 153 case "": 154 // empty tag is allowed for some reason 155 case "-": 156 ts.Ignored = true 157 case "nil", "nilString", "nilList": 158 ts.NilOK = true 159 if field.Type.Kind != reflect.Ptr { 160 return ts, TagError{Field: name, Tag: t, Err: "field is not a pointer"} 161 } 162 switch t { 163 case "nil": 164 ts.NilKind = field.Type.Elem.DefaultNilValue() 165 case "nilString": 166 ts.NilKind = NilKindString 167 case "nilList": 168 ts.NilKind = NilKindList 169 } 170 case "optional": 171 ts.Optional = true 172 if ts.Tail { 173 return ts, TagError{Field: name, Tag: t, Err: `also has "tail" tag`} 174 } 175 case "tail": 176 ts.Tail = true 177 if field.Index != lastPublic { 178 return ts, TagError{Field: name, Tag: t, Err: "must be on last field"} 179 } 180 if ts.Optional { 181 return ts, TagError{Field: name, Tag: t, Err: `also has "optional" tag`} 182 } 183 if field.Type.Kind != reflect.Slice { 184 return ts, TagError{Field: name, Tag: t, Err: "field type is not slice"} 185 } 186 default: 187 return ts, TagError{Field: name, Tag: t, Err: "unknown tag"} 188 } 189 } 190 return ts, nil 191 } 192 193 func lastPublicField(fields []Field) int { 194 last := 0 195 for _, f := range fields { 196 if f.Exported { 197 last = f.Index 198 } 199 } 200 return last 201 } 202 203 func isUint(k reflect.Kind) bool { 204 return k >= reflect.Uint && k <= reflect.Uintptr 205 } 206 207 func isByte(typ Type) bool { 208 return typ.Kind == reflect.Uint8 && !typ.IsEncoder 209 } 210 211 func isByteArray(typ Type) bool { 212 return (typ.Kind == reflect.Slice || typ.Kind == reflect.Array) && isByte(*typ.Elem) 213 }