blob: 0a08e4f0d418d407e8c24893b824bcdc5fad7560 [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
5package jsonpb
6
7import (
8 "encoding/base64"
9 "fmt"
10 "math"
11 "strconv"
12 "strings"
13
14 "github.com/golang/protobuf/v2/internal/encoding/json"
15 "github.com/golang/protobuf/v2/internal/errors"
Herbie Onge52379a2019-03-15 18:00:19 -070016 "github.com/golang/protobuf/v2/internal/pragma"
Herbie Ongc96a79d2019-03-08 10:49:17 -080017 "github.com/golang/protobuf/v2/internal/set"
18 "github.com/golang/protobuf/v2/proto"
19 pref "github.com/golang/protobuf/v2/reflect/protoreflect"
Herbie Onge52379a2019-03-15 18:00:19 -070020 "github.com/golang/protobuf/v2/reflect/protoregistry"
Herbie Ongc96a79d2019-03-08 10:49:17 -080021)
22
23// Unmarshal reads the given []byte into the given proto.Message.
24func Unmarshal(m proto.Message, b []byte) error {
25 return UnmarshalOptions{}.Unmarshal(m, b)
26}
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.
51func (o UnmarshalOptions) Unmarshal(m proto.Message, b []byte) error {
52 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
Herbie Onge63c4c42019-03-22 22:20:22 -0700138 if isCustomType(m.Type().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
163 msgType := m.Type()
164 knownFields := m.KnownFields()
165 fieldDescs := msgType.Fields()
166 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 }
211 fd = xt
212 } else {
213 // The name can either be the JSON name or the proto field name.
214 fd = fieldDescs.ByJSONName(name)
215 if fd == nil {
216 fd = fieldDescs.ByName(pref.Name(name))
217 }
Herbie Ongc96a79d2019-03-08 10:49:17 -0800218 }
219
220 if fd == nil {
221 // Field is unknown.
Herbie Ong4f0be712019-04-25 17:57:12 -0700222 if o.DiscardUnknown {
223 if err := skipJSONValue(o.decoder); !nerr.Merge(err) {
224 return err
225 }
226 continue
227 }
Herbie Ongc96a79d2019-03-08 10:49:17 -0800228 return newError("%v contains unknown field %s", msgType.FullName(), jval)
229 }
230
231 // Do not allow duplicate fields.
232 num := uint64(fd.Number())
233 if seenNums.Has(num) {
234 return newError("%v contains repeated field %s", msgType.FullName(), jval)
235 }
236 seenNums.Set(num)
237
Herbie Onge63c4c42019-03-22 22:20:22 -0700238 // No need to set values for JSON null unless the field type is
Herbie Ong300b9fe2019-03-29 15:42:20 -0700239 // google.protobuf.Value or google.protobuf.NullValue.
240 if o.decoder.Peek() == json.Null && !isKnownValue(fd) && !isNullValue(fd) {
Herbie Ong822de2d2019-03-27 13:16:23 -0700241 o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800242 continue
243 }
244
245 if cardinality := fd.Cardinality(); cardinality == pref.Repeated {
246 // Map or list fields have cardinality of repeated.
Herbie Ong822de2d2019-03-27 13:16:23 -0700247 if err := o.unmarshalRepeated(knownFields, fd); !nerr.Merge(err) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800248 return errors.New("%v|%q: %v", fd.FullName(), name, err)
249 }
250 } else {
Herbie Ong8a1d4602019-04-02 20:19:36 -0700251 // If field is a oneof, check if it has already been set.
Joe Tsaid24bc722019-04-15 23:39:09 -0700252 if od := fd.Oneof(); od != nil {
Herbie Ong8a1d4602019-04-02 20:19:36 -0700253 idx := uint64(od.Index())
254 if seenOneofs.Has(idx) {
255 return errors.New("%v: oneof is already set", od.FullName())
256 }
257 seenOneofs.Set(idx)
258 }
259
Herbie Ongc96a79d2019-03-08 10:49:17 -0800260 // Required or optional fields.
Herbie Ong822de2d2019-03-27 13:16:23 -0700261 if err := o.unmarshalSingular(knownFields, fd); !nerr.Merge(err) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800262 return errors.New("%v|%q: %v", fd.FullName(), name, err)
263 }
Herbie Ongc96a79d2019-03-08 10:49:17 -0800264 }
265 }
266
267 return nerr.E
268}
269
Herbie Onge52379a2019-03-15 18:00:19 -0700270// findExtension returns protoreflect.ExtensionType from the resolver if found.
Herbie Ong822de2d2019-03-27 13:16:23 -0700271func (o UnmarshalOptions) findExtension(xtName pref.FullName) (pref.ExtensionType, error) {
272 xt, err := o.Resolver.FindExtensionByName(xtName)
Herbie Onge52379a2019-03-15 18:00:19 -0700273 if err == nil {
274 return xt, nil
275 }
276
277 // Check if this is a MessageSet extension field.
Herbie Ong822de2d2019-03-27 13:16:23 -0700278 xt, err = o.Resolver.FindExtensionByName(xtName + ".message_set_extension")
Herbie Onge52379a2019-03-15 18:00:19 -0700279 if err == nil && isMessageSetExtension(xt) {
280 return xt, nil
281 }
282 return nil, protoregistry.NotFound
283}
284
Herbie Ong300b9fe2019-03-29 15:42:20 -0700285func isKnownValue(fd pref.FieldDescriptor) bool {
Joe Tsaid24bc722019-04-15 23:39:09 -0700286 md := fd.Message()
Herbie Ong300b9fe2019-03-29 15:42:20 -0700287 return md != nil && md.FullName() == "google.protobuf.Value"
288}
289
290func isNullValue(fd pref.FieldDescriptor) bool {
Joe Tsaid24bc722019-04-15 23:39:09 -0700291 ed := fd.Enum()
Herbie Ong300b9fe2019-03-29 15:42:20 -0700292 return ed != nil && ed.FullName() == "google.protobuf.NullValue"
293}
294
Herbie Ongc96a79d2019-03-08 10:49:17 -0800295// unmarshalSingular unmarshals to the non-repeated field specified by the given
296// FieldDescriptor.
Herbie Ong822de2d2019-03-27 13:16:23 -0700297func (o UnmarshalOptions) unmarshalSingular(knownFields pref.KnownFields, fd pref.FieldDescriptor) error {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800298 var val pref.Value
299 var err error
300 num := fd.Number()
301
302 switch fd.Kind() {
303 case pref.MessageKind, pref.GroupKind:
304 m := knownFields.NewMessage(num)
Herbie Ong8ac9dd22019-03-27 12:20:50 -0700305 err = o.unmarshalMessage(m, false)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800306 val = pref.ValueOf(m)
307 default:
Herbie Ong822de2d2019-03-27 13:16:23 -0700308 val, err = o.unmarshalScalar(fd)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800309 }
310
311 var nerr errors.NonFatal
312 if !nerr.Merge(err) {
313 return err
314 }
315 knownFields.Set(num, val)
316 return nerr.E
317}
318
319// unmarshalScalar unmarshals to a scalar/enum protoreflect.Value specified by
320// the given FieldDescriptor.
Herbie Ong822de2d2019-03-27 13:16:23 -0700321func (o UnmarshalOptions) unmarshalScalar(fd pref.FieldDescriptor) (pref.Value, error) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800322 const b32 int = 32
323 const b64 int = 64
324
325 var nerr errors.NonFatal
Herbie Ong822de2d2019-03-27 13:16:23 -0700326 jval, err := o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800327 if !nerr.Merge(err) {
328 return pref.Value{}, err
329 }
330
331 kind := fd.Kind()
332 switch kind {
333 case pref.BoolKind:
334 return unmarshalBool(jval)
335
336 case pref.Int32Kind, pref.Sint32Kind, pref.Sfixed32Kind:
337 return unmarshalInt(jval, b32)
338
339 case pref.Int64Kind, pref.Sint64Kind, pref.Sfixed64Kind:
340 return unmarshalInt(jval, b64)
341
342 case pref.Uint32Kind, pref.Fixed32Kind:
343 return unmarshalUint(jval, b32)
344
345 case pref.Uint64Kind, pref.Fixed64Kind:
346 return unmarshalUint(jval, b64)
347
348 case pref.FloatKind:
349 return unmarshalFloat(jval, b32)
350
351 case pref.DoubleKind:
352 return unmarshalFloat(jval, b64)
353
354 case pref.StringKind:
355 pval, err := unmarshalString(jval)
356 if !nerr.Merge(err) {
357 return pval, err
358 }
359 return pval, nerr.E
360
361 case pref.BytesKind:
362 return unmarshalBytes(jval)
363
364 case pref.EnumKind:
365 return unmarshalEnum(jval, fd)
366 }
367
368 panic(fmt.Sprintf("invalid scalar kind %v", kind))
369}
370
371func unmarshalBool(jval json.Value) (pref.Value, error) {
372 if jval.Type() != json.Bool {
373 return pref.Value{}, unexpectedJSONError{jval}
374 }
375 b, err := jval.Bool()
376 return pref.ValueOf(b), err
377}
378
379func unmarshalInt(jval json.Value, bitSize int) (pref.Value, error) {
380 switch jval.Type() {
381 case json.Number:
382 return getInt(jval, bitSize)
383
384 case json.String:
Herbie Onge52379a2019-03-15 18:00:19 -0700385 // Decode number from string.
Herbie Ong06a6b0b2019-04-24 21:15:45 -0700386 s := strings.TrimSpace(jval.String())
387 if len(s) != len(jval.String()) {
388 return pref.Value{}, errors.New("invalid number %v", jval.Raw())
389 }
390 dec := json.NewDecoder([]byte(s))
Herbie Ongc96a79d2019-03-08 10:49:17 -0800391 var nerr errors.NonFatal
392 jval, err := dec.Read()
393 if !nerr.Merge(err) {
394 return pref.Value{}, err
395 }
396 return getInt(jval, bitSize)
397 }
398 return pref.Value{}, unexpectedJSONError{jval}
399}
400
401func getInt(jval json.Value, bitSize int) (pref.Value, error) {
402 n, err := jval.Int(bitSize)
403 if err != nil {
404 return pref.Value{}, err
405 }
406 if bitSize == 32 {
407 return pref.ValueOf(int32(n)), nil
408 }
409 return pref.ValueOf(n), nil
410}
411
412func unmarshalUint(jval json.Value, bitSize int) (pref.Value, error) {
413 switch jval.Type() {
414 case json.Number:
415 return getUint(jval, bitSize)
416
417 case json.String:
Herbie Onge52379a2019-03-15 18:00:19 -0700418 // Decode number from string.
Herbie Ong06a6b0b2019-04-24 21:15:45 -0700419 s := strings.TrimSpace(jval.String())
420 if len(s) != len(jval.String()) {
421 return pref.Value{}, errors.New("invalid number %v", jval.Raw())
422 }
423 dec := json.NewDecoder([]byte(s))
Herbie Ongc96a79d2019-03-08 10:49:17 -0800424 var nerr errors.NonFatal
425 jval, err := dec.Read()
426 if !nerr.Merge(err) {
427 return pref.Value{}, err
428 }
429 return getUint(jval, bitSize)
430 }
431 return pref.Value{}, unexpectedJSONError{jval}
432}
433
434func getUint(jval json.Value, bitSize int) (pref.Value, error) {
435 n, err := jval.Uint(bitSize)
436 if err != nil {
437 return pref.Value{}, err
438 }
439 if bitSize == 32 {
440 return pref.ValueOf(uint32(n)), nil
441 }
442 return pref.ValueOf(n), nil
443}
444
445func unmarshalFloat(jval json.Value, bitSize int) (pref.Value, error) {
446 switch jval.Type() {
447 case json.Number:
448 return getFloat(jval, bitSize)
449
450 case json.String:
451 s := jval.String()
452 switch s {
453 case "NaN":
454 if bitSize == 32 {
455 return pref.ValueOf(float32(math.NaN())), nil
456 }
457 return pref.ValueOf(math.NaN()), nil
458 case "Infinity":
459 if bitSize == 32 {
460 return pref.ValueOf(float32(math.Inf(+1))), nil
461 }
462 return pref.ValueOf(math.Inf(+1)), nil
463 case "-Infinity":
464 if bitSize == 32 {
465 return pref.ValueOf(float32(math.Inf(-1))), nil
466 }
467 return pref.ValueOf(math.Inf(-1)), nil
468 }
Herbie Onge52379a2019-03-15 18:00:19 -0700469 // Decode number from string.
Herbie Ong06a6b0b2019-04-24 21:15:45 -0700470 if len(s) != len(strings.TrimSpace(s)) {
471 return pref.Value{}, errors.New("invalid number %v", jval.Raw())
472 }
Herbie Onge52379a2019-03-15 18:00:19 -0700473 dec := json.NewDecoder([]byte(s))
Herbie Ongc96a79d2019-03-08 10:49:17 -0800474 var nerr errors.NonFatal
475 jval, err := dec.Read()
476 if !nerr.Merge(err) {
477 return pref.Value{}, err
478 }
479 return getFloat(jval, bitSize)
480 }
481 return pref.Value{}, unexpectedJSONError{jval}
482}
483
484func getFloat(jval json.Value, bitSize int) (pref.Value, error) {
485 n, err := jval.Float(bitSize)
486 if err != nil {
487 return pref.Value{}, err
488 }
489 if bitSize == 32 {
490 return pref.ValueOf(float32(n)), nil
491 }
492 return pref.ValueOf(n), nil
493}
494
495func unmarshalString(jval json.Value) (pref.Value, error) {
496 if jval.Type() != json.String {
497 return pref.Value{}, unexpectedJSONError{jval}
498 }
499 return pref.ValueOf(jval.String()), nil
500}
501
502func unmarshalBytes(jval json.Value) (pref.Value, error) {
503 if jval.Type() != json.String {
504 return pref.Value{}, unexpectedJSONError{jval}
505 }
506
507 s := jval.String()
508 enc := base64.StdEncoding
509 if strings.ContainsAny(s, "-_") {
510 enc = base64.URLEncoding
511 }
512 if len(s)%4 != 0 {
513 enc = enc.WithPadding(base64.NoPadding)
514 }
515 b, err := enc.DecodeString(s)
516 if err != nil {
517 return pref.Value{}, err
518 }
519 return pref.ValueOf(b), nil
520}
521
522func unmarshalEnum(jval json.Value, fd pref.FieldDescriptor) (pref.Value, error) {
523 switch jval.Type() {
524 case json.String:
525 // Lookup EnumNumber based on name.
526 s := jval.String()
Joe Tsaid24bc722019-04-15 23:39:09 -0700527 if enumVal := fd.Enum().Values().ByName(pref.Name(s)); enumVal != nil {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800528 return pref.ValueOf(enumVal.Number()), nil
529 }
530 return pref.Value{}, newError("invalid enum value %q", jval)
531
532 case json.Number:
533 n, err := jval.Int(32)
534 if err != nil {
535 return pref.Value{}, err
536 }
537 return pref.ValueOf(pref.EnumNumber(n)), nil
Herbie Ong300b9fe2019-03-29 15:42:20 -0700538
539 case json.Null:
540 // This is only valid for google.protobuf.NullValue.
541 if isNullValue(fd) {
542 return pref.ValueOf(pref.EnumNumber(0)), nil
543 }
Herbie Ongc96a79d2019-03-08 10:49:17 -0800544 }
545
546 return pref.Value{}, unexpectedJSONError{jval}
547}
548
549// unmarshalRepeated unmarshals into a repeated field.
Herbie Ong822de2d2019-03-27 13:16:23 -0700550func (o UnmarshalOptions) unmarshalRepeated(knownFields pref.KnownFields, fd pref.FieldDescriptor) error {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800551 var nerr errors.NonFatal
552 num := fd.Number()
553 val := knownFields.Get(num)
554 if !fd.IsMap() {
Herbie Ong822de2d2019-03-27 13:16:23 -0700555 if err := o.unmarshalList(val.List(), fd); !nerr.Merge(err) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800556 return err
557 }
558 } else {
Herbie Ong822de2d2019-03-27 13:16:23 -0700559 if err := o.unmarshalMap(val.Map(), fd); !nerr.Merge(err) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800560 return err
561 }
562 }
563 return nerr.E
564}
565
566// unmarshalList unmarshals into given protoreflect.List.
Herbie Ong822de2d2019-03-27 13:16:23 -0700567func (o UnmarshalOptions) unmarshalList(list pref.List, fd pref.FieldDescriptor) error {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800568 var nerr errors.NonFatal
Herbie Ong822de2d2019-03-27 13:16:23 -0700569 jval, err := o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800570 if !nerr.Merge(err) {
571 return err
572 }
573 if jval.Type() != json.StartArray {
574 return unexpectedJSONError{jval}
575 }
576
577 switch fd.Kind() {
578 case pref.MessageKind, pref.GroupKind:
579 for {
580 m := list.NewMessage()
Herbie Ong8ac9dd22019-03-27 12:20:50 -0700581 err := o.unmarshalMessage(m, false)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800582 if !nerr.Merge(err) {
583 if e, ok := err.(unexpectedJSONError); ok {
584 if e.value.Type() == json.EndArray {
585 // Done with list.
586 return nerr.E
587 }
588 }
589 return err
590 }
591 list.Append(pref.ValueOf(m))
592 }
593 default:
594 for {
Herbie Ong822de2d2019-03-27 13:16:23 -0700595 val, err := o.unmarshalScalar(fd)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800596 if !nerr.Merge(err) {
597 if e, ok := err.(unexpectedJSONError); ok {
598 if e.value.Type() == json.EndArray {
599 // Done with list.
600 return nerr.E
601 }
602 }
603 return err
604 }
605 list.Append(val)
606 }
607 }
608 return nerr.E
609}
610
611// unmarshalMap unmarshals into given protoreflect.Map.
Herbie Ong822de2d2019-03-27 13:16:23 -0700612func (o UnmarshalOptions) unmarshalMap(mmap pref.Map, fd pref.FieldDescriptor) error {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800613 var nerr errors.NonFatal
614
Herbie Ong822de2d2019-03-27 13:16:23 -0700615 jval, err := o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800616 if !nerr.Merge(err) {
617 return err
618 }
619 if jval.Type() != json.StartObject {
620 return unexpectedJSONError{jval}
621 }
622
Joe Tsaid24bc722019-04-15 23:39:09 -0700623 fields := fd.Message().Fields()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800624 keyDesc := fields.ByNumber(1)
625 valDesc := fields.ByNumber(2)
626
627 // Determine ahead whether map entry is a scalar type or a message type in
628 // order to call the appropriate unmarshalMapValue func inside the for loop
629 // below.
630 unmarshalMapValue := func() (pref.Value, error) {
Herbie Ong822de2d2019-03-27 13:16:23 -0700631 return o.unmarshalScalar(valDesc)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800632 }
633 switch valDesc.Kind() {
634 case pref.MessageKind, pref.GroupKind:
635 unmarshalMapValue = func() (pref.Value, error) {
Herbie Onge63c4c42019-03-22 22:20:22 -0700636 var nerr errors.NonFatal
Herbie Ongc96a79d2019-03-08 10:49:17 -0800637 m := mmap.NewMessage()
Herbie Ong8ac9dd22019-03-27 12:20:50 -0700638 if err := o.unmarshalMessage(m, false); !nerr.Merge(err) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800639 return pref.Value{}, err
640 }
Herbie Onge63c4c42019-03-22 22:20:22 -0700641 return pref.ValueOf(m), nerr.E
Herbie Ongc96a79d2019-03-08 10:49:17 -0800642 }
643 }
644
645Loop:
646 for {
647 // Read field name.
Herbie Ong822de2d2019-03-27 13:16:23 -0700648 jval, err := o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800649 if !nerr.Merge(err) {
650 return err
651 }
652 switch jval.Type() {
653 default:
654 return unexpectedJSONError{jval}
655 case json.EndObject:
656 break Loop
657 case json.Name:
658 // Continue.
659 }
660
661 name, err := jval.Name()
662 if !nerr.Merge(err) {
663 return err
664 }
665
666 // Unmarshal field name.
667 pkey, err := unmarshalMapKey(name, keyDesc)
668 if !nerr.Merge(err) {
669 return err
670 }
671
672 // Check for duplicate field name.
673 if mmap.Has(pkey) {
674 return newError("duplicate map key %q", jval)
675 }
676
677 // Read and unmarshal field value.
678 pval, err := unmarshalMapValue()
679 if !nerr.Merge(err) {
680 return err
681 }
682
683 mmap.Set(pkey, pval)
684 }
685
686 return nerr.E
687}
688
689// unmarshalMapKey converts given string into a protoreflect.MapKey. A map key type is any
690// integral or string type.
691func unmarshalMapKey(name string, fd pref.FieldDescriptor) (pref.MapKey, error) {
692 const b32 = 32
693 const b64 = 64
694 const base10 = 10
695
696 kind := fd.Kind()
697 switch kind {
698 case pref.StringKind:
699 return pref.ValueOf(name).MapKey(), nil
700
701 case pref.BoolKind:
702 switch name {
703 case "true":
704 return pref.ValueOf(true).MapKey(), nil
705 case "false":
706 return pref.ValueOf(false).MapKey(), nil
707 }
708 return pref.MapKey{}, errors.New("invalid value for boolean key %q", name)
709
710 case pref.Int32Kind, pref.Sint32Kind, pref.Sfixed32Kind:
711 n, err := strconv.ParseInt(name, base10, b32)
712 if err != nil {
713 return pref.MapKey{}, err
714 }
715 return pref.ValueOf(int32(n)).MapKey(), nil
716
717 case pref.Int64Kind, pref.Sint64Kind, pref.Sfixed64Kind:
718 n, err := strconv.ParseInt(name, base10, b64)
719 if err != nil {
720 return pref.MapKey{}, err
721 }
722 return pref.ValueOf(int64(n)).MapKey(), nil
723
724 case pref.Uint32Kind, pref.Fixed32Kind:
725 n, err := strconv.ParseUint(name, base10, b32)
726 if err != nil {
727 return pref.MapKey{}, err
728 }
729 return pref.ValueOf(uint32(n)).MapKey(), nil
730
731 case pref.Uint64Kind, pref.Fixed64Kind:
732 n, err := strconv.ParseUint(name, base10, b64)
733 if err != nil {
734 return pref.MapKey{}, err
735 }
736 return pref.ValueOf(uint64(n)).MapKey(), nil
737 }
738
739 panic(fmt.Sprintf("%s: invalid kind %s for map key", fd.FullName(), kind))
740}