| // Copyright 2018 The Go Authors. All rights reserved. | 
 | // Use of this source code is governed by a BSD-style | 
 | // license that can be found in the LICENSE file. | 
 |  | 
 | package proto_test | 
 |  | 
 | import ( | 
 | 	"bytes" | 
 | 	"fmt" | 
 | 	"reflect" | 
 | 	"testing" | 
 |  | 
 | 	"google.golang.org/protobuf/encoding/prototext" | 
 | 	"google.golang.org/protobuf/proto" | 
 | 	"google.golang.org/protobuf/reflect/protoreflect" | 
 | 	"google.golang.org/protobuf/testing/protopack" | 
 | 	"google.golang.org/protobuf/types/known/durationpb" | 
 |  | 
 | 	"google.golang.org/protobuf/internal/errors" | 
 | 	testpb "google.golang.org/protobuf/internal/testprotos/test" | 
 | 	test3pb "google.golang.org/protobuf/internal/testprotos/test3" | 
 | ) | 
 |  | 
 | func TestDecode(t *testing.T) { | 
 | 	for _, test := range testValidMessages { | 
 | 		if len(test.decodeTo) == 0 { | 
 | 			t.Errorf("%v: no test message types", test.desc) | 
 | 		} | 
 | 		for _, want := range test.decodeTo { | 
 | 			t.Run(fmt.Sprintf("%s (%T)", test.desc, want), func(t *testing.T) { | 
 | 				opts := test.unmarshalOptions | 
 | 				opts.AllowPartial = test.partial | 
 | 				wire := append(([]byte)(nil), test.wire...) | 
 | 				got := reflect.New(reflect.TypeOf(want).Elem()).Interface().(proto.Message) | 
 | 				if err := opts.Unmarshal(wire, got); err != nil { | 
 | 					t.Errorf("Unmarshal error: %v\nMessage:\n%v", err, prototext.Format(want)) | 
 | 					return | 
 | 				} | 
 |  | 
 | 				// Aliasing check: Unmarshal shouldn't modify the original wire | 
 | 				// bytes, and modifying the original wire bytes shouldn't affect | 
 | 				// the unmarshaled message. | 
 | 				if !bytes.Equal(test.wire, wire) { | 
 | 					t.Errorf("Unmarshal unexpectedly modified its input") | 
 | 				} | 
 | 				for i := range wire { | 
 | 					wire[i] = 0 | 
 | 				} | 
 | 				if !proto.Equal(got, want) && got.ProtoReflect().IsValid() && want.ProtoReflect().IsValid() { | 
 | 					t.Errorf("Unmarshal returned unexpected result; got:\n%v\nwant:\n%v", prototext.Format(got), prototext.Format(want)) | 
 | 				} | 
 | 			}) | 
 | 		} | 
 | 	} | 
 | } | 
 |  | 
 | func TestDecodeRequiredFieldChecks(t *testing.T) { | 
 | 	for _, test := range testValidMessages { | 
 | 		if !test.partial { | 
 | 			continue | 
 | 		} | 
 | 		for _, m := range test.decodeTo { | 
 | 			t.Run(fmt.Sprintf("%s (%T)", test.desc, m), func(t *testing.T) { | 
 | 				opts := test.unmarshalOptions | 
 | 				opts.AllowPartial = false | 
 | 				got := reflect.New(reflect.TypeOf(m).Elem()).Interface().(proto.Message) | 
 | 				if err := proto.Unmarshal(test.wire, got); err == nil { | 
 | 					t.Fatalf("Unmarshal succeeded (want error)\nMessage:\n%v", prototext.Format(got)) | 
 | 				} | 
 | 			}) | 
 | 		} | 
 | 	} | 
 | } | 
 |  | 
 | func TestDecodeInvalidMessages(t *testing.T) { | 
 | 	for _, test := range testInvalidMessages { | 
 | 		if len(test.decodeTo) == 0 { | 
 | 			t.Errorf("%v: no test message types", test.desc) | 
 | 		} | 
 | 		for _, want := range test.decodeTo { | 
 | 			t.Run(fmt.Sprintf("%s (%T)", test.desc, want), func(t *testing.T) { | 
 | 				opts := test.unmarshalOptions | 
 | 				opts.AllowPartial = test.partial | 
 | 				got := want.ProtoReflect().New().Interface() | 
 | 				if err := opts.Unmarshal(test.wire, got); err == nil { | 
 | 					t.Errorf("Unmarshal unexpectedly succeeded\ninput bytes: [%x]\nMessage:\n%v", test.wire, prototext.Format(got)) | 
 | 				} else if !errors.Is(err, proto.Error) { | 
 | 					t.Errorf("Unmarshal error is not a proto.Error: %v", err) | 
 | 				} | 
 | 			}) | 
 | 		} | 
 | 	} | 
 | } | 
 |  | 
 | func TestDecodeZeroLengthBytes(t *testing.T) { | 
 | 	// Verify that proto3 bytes fields don't give the mistaken | 
 | 	// impression that they preserve presence. | 
 | 	wire := protopack.Message{ | 
 | 		protopack.Tag{94, protopack.BytesType}, protopack.Bytes(nil), | 
 | 	}.Marshal() | 
 | 	m := &test3pb.TestAllTypes{} | 
 | 	if err := proto.Unmarshal(wire, m); err != nil { | 
 | 		t.Fatal(err) | 
 | 	} | 
 | 	if m.OptionalBytes != nil { | 
 | 		t.Errorf("unmarshal zero-length proto3 bytes field: got %v, want nil", m.OptionalBytes) | 
 | 	} | 
 | } | 
 |  | 
 | func TestDecodeOneofNilWrapper(t *testing.T) { | 
 | 	wire := protopack.Message{ | 
 | 		protopack.Tag{111, protopack.VarintType}, protopack.Varint(1111), | 
 | 	}.Marshal() | 
 | 	m := &testpb.TestAllTypes{OneofField: (*testpb.TestAllTypes_OneofUint32)(nil)} | 
 | 	if err := proto.Unmarshal(wire, m); err != nil { | 
 | 		t.Fatal(err) | 
 | 	} | 
 | 	if got := m.GetOneofUint32(); got != 1111 { | 
 | 		t.Errorf("GetOneofUint32() = %v, want %v", got, 1111) | 
 | 	} | 
 | } | 
 |  | 
 | func TestDecodeEmptyBytes(t *testing.T) { | 
 | 	// There's really nothing wrong with a nil entry in a [][]byte, | 
 | 	// but we take care to produce non-nil []bytes for zero-length | 
 | 	// byte strings, so test for it. | 
 | 	m := &testpb.TestAllTypes{} | 
 | 	b := protopack.Message{ | 
 | 		protopack.Tag{45, protopack.BytesType}, protopack.Bytes(nil), | 
 | 	}.Marshal() | 
 | 	if err := proto.Unmarshal(b, m); err != nil { | 
 | 		t.Fatal(err) | 
 | 	} | 
 | 	if m.RepeatedBytes[0] == nil { | 
 | 		t.Errorf("unmarshaling repeated bytes field containing zero-length value: Got nil bytes, want non-nil") | 
 | 	} | 
 | } | 
 |  | 
 | func build(m proto.Message, opts ...buildOpt) proto.Message { | 
 | 	for _, opt := range opts { | 
 | 		opt(m) | 
 | 	} | 
 | 	return m | 
 | } | 
 |  | 
 | type buildOpt func(proto.Message) | 
 |  | 
 | func unknown(raw protoreflect.RawFields) buildOpt { | 
 | 	return func(m proto.Message) { | 
 | 		m.ProtoReflect().SetUnknown(raw) | 
 | 	} | 
 | } | 
 |  | 
 | func extend(desc protoreflect.ExtensionType, value any) buildOpt { | 
 | 	return func(m proto.Message) { | 
 | 		proto.SetExtension(m, desc, value) | 
 | 	} | 
 | } | 
 |  | 
 | // This example illustrates how to unmarshal (decode) wire format encoding into | 
 | // a Protobuf message. | 
 | func ExampleUnmarshal() { | 
 | 	// This is the wire format encoding produced by the Marshal example. | 
 | 	// Typically you would read from the network, from disk, etc. | 
 | 	b := []byte{0x10, 0x7d} | 
 |  | 
 | 	var dur durationpb.Duration | 
 | 	if err := proto.Unmarshal(b, &dur); err != nil { | 
 | 		panic(err) | 
 | 	} | 
 |  | 
 | 	fmt.Printf("Protobuf wire format decoded to duration %v\n", dur.AsDuration()) | 
 |  | 
 | 	// Output: Protobuf wire format decoded to duration 125ns | 
 | } |