blob: 99e1a8b327302cb5ba76ae00b619ab866dfbf387 [file] [log] [blame]
Herbie Ongc96a79d2019-03-08 10:49:17 -08001// Copyright 2019 The Go Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
Damien Neil5c5b5312019-05-14 12:44:37 -07005package protojson
Herbie Ongc96a79d2019-03-08 10:49:17 -08006
7import (
8 "encoding/base64"
9 "fmt"
10 "math"
11 "strconv"
12 "strings"
13
Damien Neile89e6242019-05-13 23:55:40 -070014 "google.golang.org/protobuf/internal/encoding/json"
15 "google.golang.org/protobuf/internal/errors"
16 "google.golang.org/protobuf/internal/pragma"
17 "google.golang.org/protobuf/internal/set"
18 "google.golang.org/protobuf/proto"
19 pref "google.golang.org/protobuf/reflect/protoreflect"
20 "google.golang.org/protobuf/reflect/protoregistry"
Herbie Ongc96a79d2019-03-08 10:49:17 -080021)
22
23// Unmarshal reads the given []byte into the given proto.Message.
Joe Tsaicdb77732019-05-14 16:05:06 -070024func Unmarshal(b []byte, m proto.Message) error {
25 return UnmarshalOptions{}.Unmarshal(b, m)
Herbie Ongc96a79d2019-03-08 10:49:17 -080026}
27
28// UnmarshalOptions is a configurable JSON format parser.
Herbie Onge52379a2019-03-15 18:00:19 -070029type UnmarshalOptions struct {
30 pragma.NoUnkeyedLiterals
31
Herbie Ong4f0be712019-04-25 17:57:12 -070032 // If AllowPartial is set, input for messages that will result in missing
33 // required fields will not return an error.
Herbie Ong329be5b2019-03-27 14:47:59 -070034 AllowPartial bool
35
Herbie Ong4f0be712019-04-25 17:57:12 -070036 // If DiscardUnknown is set, unknown fields are ignored.
37 DiscardUnknown bool
38
Herbie Onge52379a2019-03-15 18:00:19 -070039 // Resolver is the registry used for type lookups when unmarshaling extensions
40 // and processing Any. If Resolver is not set, unmarshaling will default to
41 // using protoregistry.GlobalTypes.
42 Resolver *protoregistry.Types
Herbie Ong822de2d2019-03-27 13:16:23 -070043
44 decoder *json.Decoder
Herbie Onge52379a2019-03-15 18:00:19 -070045}
Herbie Ongc96a79d2019-03-08 10:49:17 -080046
47// Unmarshal reads the given []byte and populates the given proto.Message using
48// options in UnmarshalOptions object. It will clear the message first before
49// setting the fields. If it returns an error, the given message may be
50// partially set.
Joe Tsaicdb77732019-05-14 16:05:06 -070051func (o UnmarshalOptions) Unmarshal(b []byte, m proto.Message) error {
Herbie Ongc96a79d2019-03-08 10:49:17 -080052 mr := m.ProtoReflect()
53 // TODO: Determine if we would like to have an option for merging or only
54 // have merging behavior. We should at least be consistent with textproto
55 // marshaling.
56 resetMessage(mr)
57
Herbie Ong822de2d2019-03-27 13:16:23 -070058 if o.Resolver == nil {
59 o.Resolver = protoregistry.GlobalTypes
Herbie Onge52379a2019-03-15 18:00:19 -070060 }
Herbie Ong822de2d2019-03-27 13:16:23 -070061 o.decoder = json.NewDecoder(b)
Herbie Onge52379a2019-03-15 18:00:19 -070062
Herbie Ongc96a79d2019-03-08 10:49:17 -080063 var nerr errors.NonFatal
Herbie Ong8ac9dd22019-03-27 12:20:50 -070064 if err := o.unmarshalMessage(mr, false); !nerr.Merge(err) {
Herbie Ongc96a79d2019-03-08 10:49:17 -080065 return err
66 }
67
68 // Check for EOF.
Herbie Ong822de2d2019-03-27 13:16:23 -070069 val, err := o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -080070 if err != nil {
71 return err
72 }
73 if val.Type() != json.EOF {
74 return unexpectedJSONError{val}
75 }
Damien Neil4686e232019-04-05 13:31:40 -070076
77 if !o.AllowPartial {
78 nerr.Merge(proto.IsInitialized(m))
79 }
Herbie Ongc96a79d2019-03-08 10:49:17 -080080 return nerr.E
81}
82
83// resetMessage clears all fields of given protoreflect.Message.
84func resetMessage(m pref.Message) {
85 knownFields := m.KnownFields()
86 knownFields.Range(func(num pref.FieldNumber, _ pref.Value) bool {
87 knownFields.Clear(num)
88 return true
89 })
90 unknownFields := m.UnknownFields()
91 unknownFields.Range(func(num pref.FieldNumber, _ pref.RawFields) bool {
92 unknownFields.Set(num, nil)
93 return true
94 })
95 extTypes := knownFields.ExtensionTypes()
96 extTypes.Range(func(xt pref.ExtensionType) bool {
97 extTypes.Remove(xt)
98 return true
99 })
100}
101
102// unexpectedJSONError is an error that contains the unexpected json.Value. This
Herbie Ong822de2d2019-03-27 13:16:23 -0700103// is returned by methods to provide callers the read json.Value that it did not
104// expect.
Herbie Ongc96a79d2019-03-08 10:49:17 -0800105// TODO: Consider moving this to internal/encoding/json for consistency with
106// errors that package returns.
107type unexpectedJSONError struct {
108 value json.Value
109}
110
111func (e unexpectedJSONError) Error() string {
112 return newError("unexpected value %s", e.value).Error()
113}
114
115// newError returns an error object. If one of the values passed in is of
116// json.Value type, it produces an error with position info.
117func newError(f string, x ...interface{}) error {
118 var hasValue bool
119 var line, column int
120 for i := 0; i < len(x); i++ {
121 if val, ok := x[i].(json.Value); ok {
122 line, column = val.Position()
123 hasValue = true
124 break
125 }
126 }
127 e := errors.New(f, x...)
128 if hasValue {
129 return errors.New("(line %d:%d): %v", line, column, e)
130 }
131 return e
132}
133
Herbie Ongc96a79d2019-03-08 10:49:17 -0800134// unmarshalMessage unmarshals a message into the given protoreflect.Message.
Herbie Ong8ac9dd22019-03-27 12:20:50 -0700135func (o UnmarshalOptions) unmarshalMessage(m pref.Message, skipTypeURL bool) error {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800136 var nerr errors.NonFatal
Herbie Ongc96a79d2019-03-08 10:49:17 -0800137
Joe Tsai0fc49f82019-05-01 12:29:25 -0700138 if isCustomType(m.Descriptor().FullName()) {
Herbie Ong822de2d2019-03-27 13:16:23 -0700139 return o.unmarshalCustomType(m)
Herbie Onge63c4c42019-03-22 22:20:22 -0700140 }
Herbie Ongc96a79d2019-03-08 10:49:17 -0800141
Herbie Ong822de2d2019-03-27 13:16:23 -0700142 jval, err := o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800143 if !nerr.Merge(err) {
144 return err
145 }
146 if jval.Type() != json.StartObject {
147 return unexpectedJSONError{jval}
148 }
149
Herbie Ong8ac9dd22019-03-27 12:20:50 -0700150 if err := o.unmarshalFields(m, skipTypeURL); !nerr.Merge(err) {
Herbie Onge63c4c42019-03-22 22:20:22 -0700151 return err
152 }
153
154 return nerr.E
155}
156
157// unmarshalFields unmarshals the fields into the given protoreflect.Message.
Herbie Ong8ac9dd22019-03-27 12:20:50 -0700158func (o UnmarshalOptions) unmarshalFields(m pref.Message, skipTypeURL bool) error {
Herbie Onge63c4c42019-03-22 22:20:22 -0700159 var nerr errors.NonFatal
Herbie Onge63c4c42019-03-22 22:20:22 -0700160 var seenNums set.Ints
Herbie Ong8a1d4602019-04-02 20:19:36 -0700161 var seenOneofs set.Ints
Herbie Onge63c4c42019-03-22 22:20:22 -0700162
Joe Tsai0fc49f82019-05-01 12:29:25 -0700163 messageDesc := m.Descriptor()
Herbie Onge63c4c42019-03-22 22:20:22 -0700164 knownFields := m.KnownFields()
Joe Tsai0fc49f82019-05-01 12:29:25 -0700165 fieldDescs := messageDesc.Fields()
Herbie Onge63c4c42019-03-22 22:20:22 -0700166 xtTypes := knownFields.ExtensionTypes()
167
Herbie Ongc96a79d2019-03-08 10:49:17 -0800168Loop:
169 for {
170 // Read field name.
Herbie Ong822de2d2019-03-27 13:16:23 -0700171 jval, err := o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800172 if !nerr.Merge(err) {
173 return err
174 }
175 switch jval.Type() {
176 default:
177 return unexpectedJSONError{jval}
178 case json.EndObject:
179 break Loop
180 case json.Name:
181 // Continue below.
182 }
183
184 name, err := jval.Name()
185 if !nerr.Merge(err) {
186 return err
187 }
Herbie Ong8ac9dd22019-03-27 12:20:50 -0700188 // Unmarshaling a non-custom embedded message in Any will contain the
189 // JSON field "@type" which should be skipped because it is not a field
190 // of the embedded message, but simply an artifact of the Any format.
191 if skipTypeURL && name == "@type" {
192 o.decoder.Read()
193 continue
194 }
Herbie Ongc96a79d2019-03-08 10:49:17 -0800195
Herbie Onge52379a2019-03-15 18:00:19 -0700196 // Get the FieldDescriptor.
197 var fd pref.FieldDescriptor
198 if strings.HasPrefix(name, "[") && strings.HasSuffix(name, "]") {
199 // Only extension names are in [name] format.
200 xtName := pref.FullName(name[1 : len(name)-1])
201 xt := xtTypes.ByName(xtName)
202 if xt == nil {
Herbie Ong822de2d2019-03-27 13:16:23 -0700203 xt, err = o.findExtension(xtName)
Herbie Onge52379a2019-03-15 18:00:19 -0700204 if err != nil && err != protoregistry.NotFound {
205 return errors.New("unable to resolve [%v]: %v", xtName, err)
206 }
207 if xt != nil {
208 xtTypes.Register(xt)
209 }
210 }
Joe Tsai0fc49f82019-05-01 12:29:25 -0700211 if xt != nil {
212 fd = xt.Descriptor()
213 }
Herbie Onge52379a2019-03-15 18:00:19 -0700214 } else {
215 // The name can either be the JSON name or the proto field name.
216 fd = fieldDescs.ByJSONName(name)
217 if fd == nil {
218 fd = fieldDescs.ByName(pref.Name(name))
219 }
Herbie Ongc96a79d2019-03-08 10:49:17 -0800220 }
221
222 if fd == nil {
223 // Field is unknown.
Herbie Ong4f0be712019-04-25 17:57:12 -0700224 if o.DiscardUnknown {
225 if err := skipJSONValue(o.decoder); !nerr.Merge(err) {
226 return err
227 }
228 continue
229 }
Joe Tsai0fc49f82019-05-01 12:29:25 -0700230 return newError("%v contains unknown field %s", messageDesc.FullName(), jval)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800231 }
232
233 // Do not allow duplicate fields.
234 num := uint64(fd.Number())
235 if seenNums.Has(num) {
Joe Tsai0fc49f82019-05-01 12:29:25 -0700236 return newError("%v contains repeated field %s", messageDesc.FullName(), jval)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800237 }
238 seenNums.Set(num)
239
Herbie Onge63c4c42019-03-22 22:20:22 -0700240 // No need to set values for JSON null unless the field type is
Herbie Ong300b9fe2019-03-29 15:42:20 -0700241 // google.protobuf.Value or google.protobuf.NullValue.
242 if o.decoder.Peek() == json.Null && !isKnownValue(fd) && !isNullValue(fd) {
Herbie Ong822de2d2019-03-27 13:16:23 -0700243 o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800244 continue
245 }
246
Joe Tsaiac31a352019-05-13 14:32:56 -0700247 switch {
248 case fd.IsList():
249 list := knownFields.Get(fd.Number()).List()
250 if err := o.unmarshalList(list, fd); !nerr.Merge(err) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800251 return errors.New("%v|%q: %v", fd.FullName(), name, err)
252 }
Joe Tsaiac31a352019-05-13 14:32:56 -0700253 case fd.IsMap():
254 mmap := knownFields.Get(fd.Number()).Map()
255 if err := o.unmarshalMap(mmap, fd); !nerr.Merge(err) {
256 return errors.New("%v|%q: %v", fd.FullName(), name, err)
257 }
258 default:
Herbie Ong8a1d4602019-04-02 20:19:36 -0700259 // If field is a oneof, check if it has already been set.
Joe Tsaiac31a352019-05-13 14:32:56 -0700260 if od := fd.ContainingOneof(); od != nil {
Herbie Ong8a1d4602019-04-02 20:19:36 -0700261 idx := uint64(od.Index())
262 if seenOneofs.Has(idx) {
263 return errors.New("%v: oneof is already set", od.FullName())
264 }
265 seenOneofs.Set(idx)
266 }
267
Herbie Ongc96a79d2019-03-08 10:49:17 -0800268 // Required or optional fields.
Herbie Ong822de2d2019-03-27 13:16:23 -0700269 if err := o.unmarshalSingular(knownFields, fd); !nerr.Merge(err) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800270 return errors.New("%v|%q: %v", fd.FullName(), name, err)
271 }
Herbie Ongc96a79d2019-03-08 10:49:17 -0800272 }
273 }
274
275 return nerr.E
276}
277
Herbie Onge52379a2019-03-15 18:00:19 -0700278// findExtension returns protoreflect.ExtensionType from the resolver if found.
Herbie Ong822de2d2019-03-27 13:16:23 -0700279func (o UnmarshalOptions) findExtension(xtName pref.FullName) (pref.ExtensionType, error) {
280 xt, err := o.Resolver.FindExtensionByName(xtName)
Herbie Onge52379a2019-03-15 18:00:19 -0700281 if err == nil {
282 return xt, nil
283 }
284
285 // Check if this is a MessageSet extension field.
Herbie Ong822de2d2019-03-27 13:16:23 -0700286 xt, err = o.Resolver.FindExtensionByName(xtName + ".message_set_extension")
Herbie Onge52379a2019-03-15 18:00:19 -0700287 if err == nil && isMessageSetExtension(xt) {
288 return xt, nil
289 }
290 return nil, protoregistry.NotFound
291}
292
Herbie Ong300b9fe2019-03-29 15:42:20 -0700293func isKnownValue(fd pref.FieldDescriptor) bool {
Joe Tsaid24bc722019-04-15 23:39:09 -0700294 md := fd.Message()
Herbie Ong300b9fe2019-03-29 15:42:20 -0700295 return md != nil && md.FullName() == "google.protobuf.Value"
296}
297
298func isNullValue(fd pref.FieldDescriptor) bool {
Joe Tsaid24bc722019-04-15 23:39:09 -0700299 ed := fd.Enum()
Herbie Ong300b9fe2019-03-29 15:42:20 -0700300 return ed != nil && ed.FullName() == "google.protobuf.NullValue"
301}
302
Herbie Ongc96a79d2019-03-08 10:49:17 -0800303// unmarshalSingular unmarshals to the non-repeated field specified by the given
304// FieldDescriptor.
Herbie Ong822de2d2019-03-27 13:16:23 -0700305func (o UnmarshalOptions) unmarshalSingular(knownFields pref.KnownFields, fd pref.FieldDescriptor) error {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800306 var val pref.Value
307 var err error
308 num := fd.Number()
309
310 switch fd.Kind() {
311 case pref.MessageKind, pref.GroupKind:
312 m := knownFields.NewMessage(num)
Herbie Ong8ac9dd22019-03-27 12:20:50 -0700313 err = o.unmarshalMessage(m, false)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800314 val = pref.ValueOf(m)
315 default:
Herbie Ong822de2d2019-03-27 13:16:23 -0700316 val, err = o.unmarshalScalar(fd)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800317 }
318
319 var nerr errors.NonFatal
320 if !nerr.Merge(err) {
321 return err
322 }
323 knownFields.Set(num, val)
324 return nerr.E
325}
326
327// unmarshalScalar unmarshals to a scalar/enum protoreflect.Value specified by
328// the given FieldDescriptor.
Herbie Ong822de2d2019-03-27 13:16:23 -0700329func (o UnmarshalOptions) unmarshalScalar(fd pref.FieldDescriptor) (pref.Value, error) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800330 const b32 int = 32
331 const b64 int = 64
332
333 var nerr errors.NonFatal
Herbie Ong822de2d2019-03-27 13:16:23 -0700334 jval, err := o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800335 if !nerr.Merge(err) {
336 return pref.Value{}, err
337 }
338
339 kind := fd.Kind()
340 switch kind {
341 case pref.BoolKind:
342 return unmarshalBool(jval)
343
344 case pref.Int32Kind, pref.Sint32Kind, pref.Sfixed32Kind:
345 return unmarshalInt(jval, b32)
346
347 case pref.Int64Kind, pref.Sint64Kind, pref.Sfixed64Kind:
348 return unmarshalInt(jval, b64)
349
350 case pref.Uint32Kind, pref.Fixed32Kind:
351 return unmarshalUint(jval, b32)
352
353 case pref.Uint64Kind, pref.Fixed64Kind:
354 return unmarshalUint(jval, b64)
355
356 case pref.FloatKind:
357 return unmarshalFloat(jval, b32)
358
359 case pref.DoubleKind:
360 return unmarshalFloat(jval, b64)
361
362 case pref.StringKind:
363 pval, err := unmarshalString(jval)
364 if !nerr.Merge(err) {
365 return pval, err
366 }
367 return pval, nerr.E
368
369 case pref.BytesKind:
370 return unmarshalBytes(jval)
371
372 case pref.EnumKind:
373 return unmarshalEnum(jval, fd)
374 }
375
376 panic(fmt.Sprintf("invalid scalar kind %v", kind))
377}
378
379func unmarshalBool(jval json.Value) (pref.Value, error) {
380 if jval.Type() != json.Bool {
381 return pref.Value{}, unexpectedJSONError{jval}
382 }
383 b, err := jval.Bool()
384 return pref.ValueOf(b), err
385}
386
387func unmarshalInt(jval json.Value, bitSize int) (pref.Value, error) {
388 switch jval.Type() {
389 case json.Number:
390 return getInt(jval, bitSize)
391
392 case json.String:
Herbie Onge52379a2019-03-15 18:00:19 -0700393 // Decode number from string.
Herbie Ong06a6b0b2019-04-24 21:15:45 -0700394 s := strings.TrimSpace(jval.String())
395 if len(s) != len(jval.String()) {
396 return pref.Value{}, errors.New("invalid number %v", jval.Raw())
397 }
398 dec := json.NewDecoder([]byte(s))
Herbie Ongc96a79d2019-03-08 10:49:17 -0800399 var nerr errors.NonFatal
400 jval, err := dec.Read()
401 if !nerr.Merge(err) {
402 return pref.Value{}, err
403 }
404 return getInt(jval, bitSize)
405 }
406 return pref.Value{}, unexpectedJSONError{jval}
407}
408
409func getInt(jval json.Value, bitSize int) (pref.Value, error) {
410 n, err := jval.Int(bitSize)
411 if err != nil {
412 return pref.Value{}, err
413 }
414 if bitSize == 32 {
415 return pref.ValueOf(int32(n)), nil
416 }
417 return pref.ValueOf(n), nil
418}
419
420func unmarshalUint(jval json.Value, bitSize int) (pref.Value, error) {
421 switch jval.Type() {
422 case json.Number:
423 return getUint(jval, bitSize)
424
425 case json.String:
Herbie Onge52379a2019-03-15 18:00:19 -0700426 // Decode number from string.
Herbie Ong06a6b0b2019-04-24 21:15:45 -0700427 s := strings.TrimSpace(jval.String())
428 if len(s) != len(jval.String()) {
429 return pref.Value{}, errors.New("invalid number %v", jval.Raw())
430 }
431 dec := json.NewDecoder([]byte(s))
Herbie Ongc96a79d2019-03-08 10:49:17 -0800432 var nerr errors.NonFatal
433 jval, err := dec.Read()
434 if !nerr.Merge(err) {
435 return pref.Value{}, err
436 }
437 return getUint(jval, bitSize)
438 }
439 return pref.Value{}, unexpectedJSONError{jval}
440}
441
442func getUint(jval json.Value, bitSize int) (pref.Value, error) {
443 n, err := jval.Uint(bitSize)
444 if err != nil {
445 return pref.Value{}, err
446 }
447 if bitSize == 32 {
448 return pref.ValueOf(uint32(n)), nil
449 }
450 return pref.ValueOf(n), nil
451}
452
453func unmarshalFloat(jval json.Value, bitSize int) (pref.Value, error) {
454 switch jval.Type() {
455 case json.Number:
456 return getFloat(jval, bitSize)
457
458 case json.String:
459 s := jval.String()
460 switch s {
461 case "NaN":
462 if bitSize == 32 {
463 return pref.ValueOf(float32(math.NaN())), nil
464 }
465 return pref.ValueOf(math.NaN()), nil
466 case "Infinity":
467 if bitSize == 32 {
468 return pref.ValueOf(float32(math.Inf(+1))), nil
469 }
470 return pref.ValueOf(math.Inf(+1)), nil
471 case "-Infinity":
472 if bitSize == 32 {
473 return pref.ValueOf(float32(math.Inf(-1))), nil
474 }
475 return pref.ValueOf(math.Inf(-1)), nil
476 }
Herbie Onge52379a2019-03-15 18:00:19 -0700477 // Decode number from string.
Herbie Ong06a6b0b2019-04-24 21:15:45 -0700478 if len(s) != len(strings.TrimSpace(s)) {
479 return pref.Value{}, errors.New("invalid number %v", jval.Raw())
480 }
Herbie Onge52379a2019-03-15 18:00:19 -0700481 dec := json.NewDecoder([]byte(s))
Herbie Ongc96a79d2019-03-08 10:49:17 -0800482 var nerr errors.NonFatal
483 jval, err := dec.Read()
484 if !nerr.Merge(err) {
485 return pref.Value{}, err
486 }
487 return getFloat(jval, bitSize)
488 }
489 return pref.Value{}, unexpectedJSONError{jval}
490}
491
492func getFloat(jval json.Value, bitSize int) (pref.Value, error) {
493 n, err := jval.Float(bitSize)
494 if err != nil {
495 return pref.Value{}, err
496 }
497 if bitSize == 32 {
498 return pref.ValueOf(float32(n)), nil
499 }
500 return pref.ValueOf(n), nil
501}
502
503func unmarshalString(jval json.Value) (pref.Value, error) {
504 if jval.Type() != json.String {
505 return pref.Value{}, unexpectedJSONError{jval}
506 }
507 return pref.ValueOf(jval.String()), nil
508}
509
510func unmarshalBytes(jval json.Value) (pref.Value, error) {
511 if jval.Type() != json.String {
512 return pref.Value{}, unexpectedJSONError{jval}
513 }
514
515 s := jval.String()
516 enc := base64.StdEncoding
517 if strings.ContainsAny(s, "-_") {
518 enc = base64.URLEncoding
519 }
520 if len(s)%4 != 0 {
521 enc = enc.WithPadding(base64.NoPadding)
522 }
523 b, err := enc.DecodeString(s)
524 if err != nil {
525 return pref.Value{}, err
526 }
527 return pref.ValueOf(b), nil
528}
529
530func unmarshalEnum(jval json.Value, fd pref.FieldDescriptor) (pref.Value, error) {
531 switch jval.Type() {
532 case json.String:
533 // Lookup EnumNumber based on name.
534 s := jval.String()
Joe Tsaid24bc722019-04-15 23:39:09 -0700535 if enumVal := fd.Enum().Values().ByName(pref.Name(s)); enumVal != nil {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800536 return pref.ValueOf(enumVal.Number()), nil
537 }
538 return pref.Value{}, newError("invalid enum value %q", jval)
539
540 case json.Number:
541 n, err := jval.Int(32)
542 if err != nil {
543 return pref.Value{}, err
544 }
545 return pref.ValueOf(pref.EnumNumber(n)), nil
Herbie Ong300b9fe2019-03-29 15:42:20 -0700546
547 case json.Null:
548 // This is only valid for google.protobuf.NullValue.
549 if isNullValue(fd) {
550 return pref.ValueOf(pref.EnumNumber(0)), nil
551 }
Herbie Ongc96a79d2019-03-08 10:49:17 -0800552 }
553
554 return pref.Value{}, unexpectedJSONError{jval}
555}
556
Herbie Ong822de2d2019-03-27 13:16:23 -0700557func (o UnmarshalOptions) unmarshalList(list pref.List, fd pref.FieldDescriptor) error {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800558 var nerr errors.NonFatal
Herbie Ong822de2d2019-03-27 13:16:23 -0700559 jval, err := o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800560 if !nerr.Merge(err) {
561 return err
562 }
563 if jval.Type() != json.StartArray {
564 return unexpectedJSONError{jval}
565 }
566
567 switch fd.Kind() {
568 case pref.MessageKind, pref.GroupKind:
569 for {
570 m := list.NewMessage()
Herbie Ong8ac9dd22019-03-27 12:20:50 -0700571 err := o.unmarshalMessage(m, false)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800572 if !nerr.Merge(err) {
573 if e, ok := err.(unexpectedJSONError); ok {
574 if e.value.Type() == json.EndArray {
575 // Done with list.
576 return nerr.E
577 }
578 }
579 return err
580 }
581 list.Append(pref.ValueOf(m))
582 }
583 default:
584 for {
Herbie Ong822de2d2019-03-27 13:16:23 -0700585 val, err := o.unmarshalScalar(fd)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800586 if !nerr.Merge(err) {
587 if e, ok := err.(unexpectedJSONError); ok {
588 if e.value.Type() == json.EndArray {
589 // Done with list.
590 return nerr.E
591 }
592 }
593 return err
594 }
595 list.Append(val)
596 }
597 }
598 return nerr.E
599}
600
Herbie Ong822de2d2019-03-27 13:16:23 -0700601func (o UnmarshalOptions) unmarshalMap(mmap pref.Map, fd pref.FieldDescriptor) error {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800602 var nerr errors.NonFatal
Herbie Ong822de2d2019-03-27 13:16:23 -0700603 jval, err := o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800604 if !nerr.Merge(err) {
605 return err
606 }
607 if jval.Type() != json.StartObject {
608 return unexpectedJSONError{jval}
609 }
610
Herbie Ongc96a79d2019-03-08 10:49:17 -0800611 // Determine ahead whether map entry is a scalar type or a message type in
612 // order to call the appropriate unmarshalMapValue func inside the for loop
613 // below.
Joe Tsaiac31a352019-05-13 14:32:56 -0700614 var unmarshalMapValue func() (pref.Value, error)
615 switch fd.MapValue().Kind() {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800616 case pref.MessageKind, pref.GroupKind:
617 unmarshalMapValue = func() (pref.Value, error) {
Herbie Onge63c4c42019-03-22 22:20:22 -0700618 var nerr errors.NonFatal
Herbie Ongc96a79d2019-03-08 10:49:17 -0800619 m := mmap.NewMessage()
Herbie Ong8ac9dd22019-03-27 12:20:50 -0700620 if err := o.unmarshalMessage(m, false); !nerr.Merge(err) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800621 return pref.Value{}, err
622 }
Herbie Onge63c4c42019-03-22 22:20:22 -0700623 return pref.ValueOf(m), nerr.E
Herbie Ongc96a79d2019-03-08 10:49:17 -0800624 }
Joe Tsaiac31a352019-05-13 14:32:56 -0700625 default:
626 unmarshalMapValue = func() (pref.Value, error) {
627 return o.unmarshalScalar(fd.MapValue())
628 }
Herbie Ongc96a79d2019-03-08 10:49:17 -0800629 }
630
631Loop:
632 for {
633 // Read field name.
Herbie Ong822de2d2019-03-27 13:16:23 -0700634 jval, err := o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800635 if !nerr.Merge(err) {
636 return err
637 }
638 switch jval.Type() {
639 default:
640 return unexpectedJSONError{jval}
641 case json.EndObject:
642 break Loop
643 case json.Name:
644 // Continue.
645 }
646
647 name, err := jval.Name()
648 if !nerr.Merge(err) {
649 return err
650 }
651
652 // Unmarshal field name.
Joe Tsaiac31a352019-05-13 14:32:56 -0700653 pkey, err := unmarshalMapKey(name, fd.MapKey())
Herbie Ongc96a79d2019-03-08 10:49:17 -0800654 if !nerr.Merge(err) {
655 return err
656 }
657
658 // Check for duplicate field name.
659 if mmap.Has(pkey) {
660 return newError("duplicate map key %q", jval)
661 }
662
663 // Read and unmarshal field value.
664 pval, err := unmarshalMapValue()
665 if !nerr.Merge(err) {
666 return err
667 }
668
669 mmap.Set(pkey, pval)
670 }
671
672 return nerr.E
673}
674
675// unmarshalMapKey converts given string into a protoreflect.MapKey. A map key type is any
676// integral or string type.
677func unmarshalMapKey(name string, fd pref.FieldDescriptor) (pref.MapKey, error) {
678 const b32 = 32
679 const b64 = 64
680 const base10 = 10
681
682 kind := fd.Kind()
683 switch kind {
684 case pref.StringKind:
685 return pref.ValueOf(name).MapKey(), nil
686
687 case pref.BoolKind:
688 switch name {
689 case "true":
690 return pref.ValueOf(true).MapKey(), nil
691 case "false":
692 return pref.ValueOf(false).MapKey(), nil
693 }
694 return pref.MapKey{}, errors.New("invalid value for boolean key %q", name)
695
696 case pref.Int32Kind, pref.Sint32Kind, pref.Sfixed32Kind:
697 n, err := strconv.ParseInt(name, base10, b32)
698 if err != nil {
699 return pref.MapKey{}, err
700 }
701 return pref.ValueOf(int32(n)).MapKey(), nil
702
703 case pref.Int64Kind, pref.Sint64Kind, pref.Sfixed64Kind:
704 n, err := strconv.ParseInt(name, base10, b64)
705 if err != nil {
706 return pref.MapKey{}, err
707 }
708 return pref.ValueOf(int64(n)).MapKey(), nil
709
710 case pref.Uint32Kind, pref.Fixed32Kind:
711 n, err := strconv.ParseUint(name, base10, b32)
712 if err != nil {
713 return pref.MapKey{}, err
714 }
715 return pref.ValueOf(uint32(n)).MapKey(), nil
716
717 case pref.Uint64Kind, pref.Fixed64Kind:
718 n, err := strconv.ParseUint(name, base10, b64)
719 if err != nil {
720 return pref.MapKey{}, err
721 }
722 return pref.ValueOf(uint64(n)).MapKey(), nil
723 }
724
725 panic(fmt.Sprintf("%s: invalid kind %s for map key", fd.FullName(), kind))
726}