blob: 7e142c5c7b1dc596d30e51ba8294bcc6830af45b [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
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
247 if cardinality := fd.Cardinality(); cardinality == pref.Repeated {
248 // Map or list fields have cardinality of repeated.
Herbie Ong822de2d2019-03-27 13:16:23 -0700249 if err := o.unmarshalRepeated(knownFields, fd); !nerr.Merge(err) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800250 return errors.New("%v|%q: %v", fd.FullName(), name, err)
251 }
252 } else {
Herbie Ong8a1d4602019-04-02 20:19:36 -0700253 // If field is a oneof, check if it has already been set.
Joe Tsaid24bc722019-04-15 23:39:09 -0700254 if od := fd.Oneof(); od != nil {
Herbie Ong8a1d4602019-04-02 20:19:36 -0700255 idx := uint64(od.Index())
256 if seenOneofs.Has(idx) {
257 return errors.New("%v: oneof is already set", od.FullName())
258 }
259 seenOneofs.Set(idx)
260 }
261
Herbie Ongc96a79d2019-03-08 10:49:17 -0800262 // Required or optional fields.
Herbie Ong822de2d2019-03-27 13:16:23 -0700263 if err := o.unmarshalSingular(knownFields, fd); !nerr.Merge(err) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800264 return errors.New("%v|%q: %v", fd.FullName(), name, err)
265 }
Herbie Ongc96a79d2019-03-08 10:49:17 -0800266 }
267 }
268
269 return nerr.E
270}
271
Herbie Onge52379a2019-03-15 18:00:19 -0700272// findExtension returns protoreflect.ExtensionType from the resolver if found.
Herbie Ong822de2d2019-03-27 13:16:23 -0700273func (o UnmarshalOptions) findExtension(xtName pref.FullName) (pref.ExtensionType, error) {
274 xt, err := o.Resolver.FindExtensionByName(xtName)
Herbie Onge52379a2019-03-15 18:00:19 -0700275 if err == nil {
276 return xt, nil
277 }
278
279 // Check if this is a MessageSet extension field.
Herbie Ong822de2d2019-03-27 13:16:23 -0700280 xt, err = o.Resolver.FindExtensionByName(xtName + ".message_set_extension")
Herbie Onge52379a2019-03-15 18:00:19 -0700281 if err == nil && isMessageSetExtension(xt) {
282 return xt, nil
283 }
284 return nil, protoregistry.NotFound
285}
286
Herbie Ong300b9fe2019-03-29 15:42:20 -0700287func isKnownValue(fd pref.FieldDescriptor) bool {
Joe Tsaid24bc722019-04-15 23:39:09 -0700288 md := fd.Message()
Herbie Ong300b9fe2019-03-29 15:42:20 -0700289 return md != nil && md.FullName() == "google.protobuf.Value"
290}
291
292func isNullValue(fd pref.FieldDescriptor) bool {
Joe Tsaid24bc722019-04-15 23:39:09 -0700293 ed := fd.Enum()
Herbie Ong300b9fe2019-03-29 15:42:20 -0700294 return ed != nil && ed.FullName() == "google.protobuf.NullValue"
295}
296
Herbie Ongc96a79d2019-03-08 10:49:17 -0800297// unmarshalSingular unmarshals to the non-repeated field specified by the given
298// FieldDescriptor.
Herbie Ong822de2d2019-03-27 13:16:23 -0700299func (o UnmarshalOptions) unmarshalSingular(knownFields pref.KnownFields, fd pref.FieldDescriptor) error {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800300 var val pref.Value
301 var err error
302 num := fd.Number()
303
304 switch fd.Kind() {
305 case pref.MessageKind, pref.GroupKind:
306 m := knownFields.NewMessage(num)
Herbie Ong8ac9dd22019-03-27 12:20:50 -0700307 err = o.unmarshalMessage(m, false)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800308 val = pref.ValueOf(m)
309 default:
Herbie Ong822de2d2019-03-27 13:16:23 -0700310 val, err = o.unmarshalScalar(fd)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800311 }
312
313 var nerr errors.NonFatal
314 if !nerr.Merge(err) {
315 return err
316 }
317 knownFields.Set(num, val)
318 return nerr.E
319}
320
321// unmarshalScalar unmarshals to a scalar/enum protoreflect.Value specified by
322// the given FieldDescriptor.
Herbie Ong822de2d2019-03-27 13:16:23 -0700323func (o UnmarshalOptions) unmarshalScalar(fd pref.FieldDescriptor) (pref.Value, error) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800324 const b32 int = 32
325 const b64 int = 64
326
327 var nerr errors.NonFatal
Herbie Ong822de2d2019-03-27 13:16:23 -0700328 jval, err := o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800329 if !nerr.Merge(err) {
330 return pref.Value{}, err
331 }
332
333 kind := fd.Kind()
334 switch kind {
335 case pref.BoolKind:
336 return unmarshalBool(jval)
337
338 case pref.Int32Kind, pref.Sint32Kind, pref.Sfixed32Kind:
339 return unmarshalInt(jval, b32)
340
341 case pref.Int64Kind, pref.Sint64Kind, pref.Sfixed64Kind:
342 return unmarshalInt(jval, b64)
343
344 case pref.Uint32Kind, pref.Fixed32Kind:
345 return unmarshalUint(jval, b32)
346
347 case pref.Uint64Kind, pref.Fixed64Kind:
348 return unmarshalUint(jval, b64)
349
350 case pref.FloatKind:
351 return unmarshalFloat(jval, b32)
352
353 case pref.DoubleKind:
354 return unmarshalFloat(jval, b64)
355
356 case pref.StringKind:
357 pval, err := unmarshalString(jval)
358 if !nerr.Merge(err) {
359 return pval, err
360 }
361 return pval, nerr.E
362
363 case pref.BytesKind:
364 return unmarshalBytes(jval)
365
366 case pref.EnumKind:
367 return unmarshalEnum(jval, fd)
368 }
369
370 panic(fmt.Sprintf("invalid scalar kind %v", kind))
371}
372
373func unmarshalBool(jval json.Value) (pref.Value, error) {
374 if jval.Type() != json.Bool {
375 return pref.Value{}, unexpectedJSONError{jval}
376 }
377 b, err := jval.Bool()
378 return pref.ValueOf(b), err
379}
380
381func unmarshalInt(jval json.Value, bitSize int) (pref.Value, error) {
382 switch jval.Type() {
383 case json.Number:
384 return getInt(jval, bitSize)
385
386 case json.String:
Herbie Onge52379a2019-03-15 18:00:19 -0700387 // Decode number from string.
Herbie Ong06a6b0b2019-04-24 21:15:45 -0700388 s := strings.TrimSpace(jval.String())
389 if len(s) != len(jval.String()) {
390 return pref.Value{}, errors.New("invalid number %v", jval.Raw())
391 }
392 dec := json.NewDecoder([]byte(s))
Herbie Ongc96a79d2019-03-08 10:49:17 -0800393 var nerr errors.NonFatal
394 jval, err := dec.Read()
395 if !nerr.Merge(err) {
396 return pref.Value{}, err
397 }
398 return getInt(jval, bitSize)
399 }
400 return pref.Value{}, unexpectedJSONError{jval}
401}
402
403func getInt(jval json.Value, bitSize int) (pref.Value, error) {
404 n, err := jval.Int(bitSize)
405 if err != nil {
406 return pref.Value{}, err
407 }
408 if bitSize == 32 {
409 return pref.ValueOf(int32(n)), nil
410 }
411 return pref.ValueOf(n), nil
412}
413
414func unmarshalUint(jval json.Value, bitSize int) (pref.Value, error) {
415 switch jval.Type() {
416 case json.Number:
417 return getUint(jval, bitSize)
418
419 case json.String:
Herbie Onge52379a2019-03-15 18:00:19 -0700420 // Decode number from string.
Herbie Ong06a6b0b2019-04-24 21:15:45 -0700421 s := strings.TrimSpace(jval.String())
422 if len(s) != len(jval.String()) {
423 return pref.Value{}, errors.New("invalid number %v", jval.Raw())
424 }
425 dec := json.NewDecoder([]byte(s))
Herbie Ongc96a79d2019-03-08 10:49:17 -0800426 var nerr errors.NonFatal
427 jval, err := dec.Read()
428 if !nerr.Merge(err) {
429 return pref.Value{}, err
430 }
431 return getUint(jval, bitSize)
432 }
433 return pref.Value{}, unexpectedJSONError{jval}
434}
435
436func getUint(jval json.Value, bitSize int) (pref.Value, error) {
437 n, err := jval.Uint(bitSize)
438 if err != nil {
439 return pref.Value{}, err
440 }
441 if bitSize == 32 {
442 return pref.ValueOf(uint32(n)), nil
443 }
444 return pref.ValueOf(n), nil
445}
446
447func unmarshalFloat(jval json.Value, bitSize int) (pref.Value, error) {
448 switch jval.Type() {
449 case json.Number:
450 return getFloat(jval, bitSize)
451
452 case json.String:
453 s := jval.String()
454 switch s {
455 case "NaN":
456 if bitSize == 32 {
457 return pref.ValueOf(float32(math.NaN())), nil
458 }
459 return pref.ValueOf(math.NaN()), nil
460 case "Infinity":
461 if bitSize == 32 {
462 return pref.ValueOf(float32(math.Inf(+1))), nil
463 }
464 return pref.ValueOf(math.Inf(+1)), nil
465 case "-Infinity":
466 if bitSize == 32 {
467 return pref.ValueOf(float32(math.Inf(-1))), nil
468 }
469 return pref.ValueOf(math.Inf(-1)), nil
470 }
Herbie Onge52379a2019-03-15 18:00:19 -0700471 // Decode number from string.
Herbie Ong06a6b0b2019-04-24 21:15:45 -0700472 if len(s) != len(strings.TrimSpace(s)) {
473 return pref.Value{}, errors.New("invalid number %v", jval.Raw())
474 }
Herbie Onge52379a2019-03-15 18:00:19 -0700475 dec := json.NewDecoder([]byte(s))
Herbie Ongc96a79d2019-03-08 10:49:17 -0800476 var nerr errors.NonFatal
477 jval, err := dec.Read()
478 if !nerr.Merge(err) {
479 return pref.Value{}, err
480 }
481 return getFloat(jval, bitSize)
482 }
483 return pref.Value{}, unexpectedJSONError{jval}
484}
485
486func getFloat(jval json.Value, bitSize int) (pref.Value, error) {
487 n, err := jval.Float(bitSize)
488 if err != nil {
489 return pref.Value{}, err
490 }
491 if bitSize == 32 {
492 return pref.ValueOf(float32(n)), nil
493 }
494 return pref.ValueOf(n), nil
495}
496
497func unmarshalString(jval json.Value) (pref.Value, error) {
498 if jval.Type() != json.String {
499 return pref.Value{}, unexpectedJSONError{jval}
500 }
501 return pref.ValueOf(jval.String()), nil
502}
503
504func unmarshalBytes(jval json.Value) (pref.Value, error) {
505 if jval.Type() != json.String {
506 return pref.Value{}, unexpectedJSONError{jval}
507 }
508
509 s := jval.String()
510 enc := base64.StdEncoding
511 if strings.ContainsAny(s, "-_") {
512 enc = base64.URLEncoding
513 }
514 if len(s)%4 != 0 {
515 enc = enc.WithPadding(base64.NoPadding)
516 }
517 b, err := enc.DecodeString(s)
518 if err != nil {
519 return pref.Value{}, err
520 }
521 return pref.ValueOf(b), nil
522}
523
524func unmarshalEnum(jval json.Value, fd pref.FieldDescriptor) (pref.Value, error) {
525 switch jval.Type() {
526 case json.String:
527 // Lookup EnumNumber based on name.
528 s := jval.String()
Joe Tsaid24bc722019-04-15 23:39:09 -0700529 if enumVal := fd.Enum().Values().ByName(pref.Name(s)); enumVal != nil {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800530 return pref.ValueOf(enumVal.Number()), nil
531 }
532 return pref.Value{}, newError("invalid enum value %q", jval)
533
534 case json.Number:
535 n, err := jval.Int(32)
536 if err != nil {
537 return pref.Value{}, err
538 }
539 return pref.ValueOf(pref.EnumNumber(n)), nil
Herbie Ong300b9fe2019-03-29 15:42:20 -0700540
541 case json.Null:
542 // This is only valid for google.protobuf.NullValue.
543 if isNullValue(fd) {
544 return pref.ValueOf(pref.EnumNumber(0)), nil
545 }
Herbie Ongc96a79d2019-03-08 10:49:17 -0800546 }
547
548 return pref.Value{}, unexpectedJSONError{jval}
549}
550
551// unmarshalRepeated unmarshals into a repeated field.
Herbie Ong822de2d2019-03-27 13:16:23 -0700552func (o UnmarshalOptions) unmarshalRepeated(knownFields pref.KnownFields, fd pref.FieldDescriptor) error {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800553 var nerr errors.NonFatal
554 num := fd.Number()
555 val := knownFields.Get(num)
556 if !fd.IsMap() {
Herbie Ong822de2d2019-03-27 13:16:23 -0700557 if err := o.unmarshalList(val.List(), fd); !nerr.Merge(err) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800558 return err
559 }
560 } else {
Herbie Ong822de2d2019-03-27 13:16:23 -0700561 if err := o.unmarshalMap(val.Map(), fd); !nerr.Merge(err) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800562 return err
563 }
564 }
565 return nerr.E
566}
567
568// unmarshalList unmarshals into given protoreflect.List.
Herbie Ong822de2d2019-03-27 13:16:23 -0700569func (o UnmarshalOptions) unmarshalList(list pref.List, fd pref.FieldDescriptor) error {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800570 var nerr errors.NonFatal
Herbie Ong822de2d2019-03-27 13:16:23 -0700571 jval, err := o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800572 if !nerr.Merge(err) {
573 return err
574 }
575 if jval.Type() != json.StartArray {
576 return unexpectedJSONError{jval}
577 }
578
579 switch fd.Kind() {
580 case pref.MessageKind, pref.GroupKind:
581 for {
582 m := list.NewMessage()
Herbie Ong8ac9dd22019-03-27 12:20:50 -0700583 err := o.unmarshalMessage(m, false)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800584 if !nerr.Merge(err) {
585 if e, ok := err.(unexpectedJSONError); ok {
586 if e.value.Type() == json.EndArray {
587 // Done with list.
588 return nerr.E
589 }
590 }
591 return err
592 }
593 list.Append(pref.ValueOf(m))
594 }
595 default:
596 for {
Herbie Ong822de2d2019-03-27 13:16:23 -0700597 val, err := o.unmarshalScalar(fd)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800598 if !nerr.Merge(err) {
599 if e, ok := err.(unexpectedJSONError); ok {
600 if e.value.Type() == json.EndArray {
601 // Done with list.
602 return nerr.E
603 }
604 }
605 return err
606 }
607 list.Append(val)
608 }
609 }
610 return nerr.E
611}
612
613// unmarshalMap unmarshals into given protoreflect.Map.
Herbie Ong822de2d2019-03-27 13:16:23 -0700614func (o UnmarshalOptions) unmarshalMap(mmap pref.Map, fd pref.FieldDescriptor) error {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800615 var nerr errors.NonFatal
616
Herbie Ong822de2d2019-03-27 13:16:23 -0700617 jval, err := o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800618 if !nerr.Merge(err) {
619 return err
620 }
621 if jval.Type() != json.StartObject {
622 return unexpectedJSONError{jval}
623 }
624
Joe Tsaid24bc722019-04-15 23:39:09 -0700625 fields := fd.Message().Fields()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800626 keyDesc := fields.ByNumber(1)
627 valDesc := fields.ByNumber(2)
628
629 // Determine ahead whether map entry is a scalar type or a message type in
630 // order to call the appropriate unmarshalMapValue func inside the for loop
631 // below.
632 unmarshalMapValue := func() (pref.Value, error) {
Herbie Ong822de2d2019-03-27 13:16:23 -0700633 return o.unmarshalScalar(valDesc)
Herbie Ongc96a79d2019-03-08 10:49:17 -0800634 }
635 switch valDesc.Kind() {
636 case pref.MessageKind, pref.GroupKind:
637 unmarshalMapValue = func() (pref.Value, error) {
Herbie Onge63c4c42019-03-22 22:20:22 -0700638 var nerr errors.NonFatal
Herbie Ongc96a79d2019-03-08 10:49:17 -0800639 m := mmap.NewMessage()
Herbie Ong8ac9dd22019-03-27 12:20:50 -0700640 if err := o.unmarshalMessage(m, false); !nerr.Merge(err) {
Herbie Ongc96a79d2019-03-08 10:49:17 -0800641 return pref.Value{}, err
642 }
Herbie Onge63c4c42019-03-22 22:20:22 -0700643 return pref.ValueOf(m), nerr.E
Herbie Ongc96a79d2019-03-08 10:49:17 -0800644 }
645 }
646
647Loop:
648 for {
649 // Read field name.
Herbie Ong822de2d2019-03-27 13:16:23 -0700650 jval, err := o.decoder.Read()
Herbie Ongc96a79d2019-03-08 10:49:17 -0800651 if !nerr.Merge(err) {
652 return err
653 }
654 switch jval.Type() {
655 default:
656 return unexpectedJSONError{jval}
657 case json.EndObject:
658 break Loop
659 case json.Name:
660 // Continue.
661 }
662
663 name, err := jval.Name()
664 if !nerr.Merge(err) {
665 return err
666 }
667
668 // Unmarshal field name.
669 pkey, err := unmarshalMapKey(name, keyDesc)
670 if !nerr.Merge(err) {
671 return err
672 }
673
674 // Check for duplicate field name.
675 if mmap.Has(pkey) {
676 return newError("duplicate map key %q", jval)
677 }
678
679 // Read and unmarshal field value.
680 pval, err := unmarshalMapValue()
681 if !nerr.Merge(err) {
682 return err
683 }
684
685 mmap.Set(pkey, pval)
686 }
687
688 return nerr.E
689}
690
691// unmarshalMapKey converts given string into a protoreflect.MapKey. A map key type is any
692// integral or string type.
693func unmarshalMapKey(name string, fd pref.FieldDescriptor) (pref.MapKey, error) {
694 const b32 = 32
695 const b64 = 64
696 const base10 = 10
697
698 kind := fd.Kind()
699 switch kind {
700 case pref.StringKind:
701 return pref.ValueOf(name).MapKey(), nil
702
703 case pref.BoolKind:
704 switch name {
705 case "true":
706 return pref.ValueOf(true).MapKey(), nil
707 case "false":
708 return pref.ValueOf(false).MapKey(), nil
709 }
710 return pref.MapKey{}, errors.New("invalid value for boolean key %q", name)
711
712 case pref.Int32Kind, pref.Sint32Kind, pref.Sfixed32Kind:
713 n, err := strconv.ParseInt(name, base10, b32)
714 if err != nil {
715 return pref.MapKey{}, err
716 }
717 return pref.ValueOf(int32(n)).MapKey(), nil
718
719 case pref.Int64Kind, pref.Sint64Kind, pref.Sfixed64Kind:
720 n, err := strconv.ParseInt(name, base10, b64)
721 if err != nil {
722 return pref.MapKey{}, err
723 }
724 return pref.ValueOf(int64(n)).MapKey(), nil
725
726 case pref.Uint32Kind, pref.Fixed32Kind:
727 n, err := strconv.ParseUint(name, base10, b32)
728 if err != nil {
729 return pref.MapKey{}, err
730 }
731 return pref.ValueOf(uint32(n)).MapKey(), nil
732
733 case pref.Uint64Kind, pref.Fixed64Kind:
734 n, err := strconv.ParseUint(name, base10, b64)
735 if err != nil {
736 return pref.MapKey{}, err
737 }
738 return pref.ValueOf(uint64(n)).MapKey(), nil
739 }
740
741 panic(fmt.Sprintf("%s: invalid kind %s for map key", fd.FullName(), kind))
742}