2019-08-02 23:58:08 +00:00
|
|
|
// Copyright 2019 The Go Authors. All rights reserved.
|
2020-02-20 18:30:38 +00:00
|
|
|
// Use of this source code is governed by a BSD-style
|
2019-08-02 23:58:08 +00:00
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package proto_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2020-01-29 22:40:11 +00:00
|
|
|
"sync"
|
2019-08-02 23:58:08 +00:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/google/go-cmp/cmp"
|
2020-02-25 20:51:10 +00:00
|
|
|
|
2019-08-02 23:58:08 +00:00
|
|
|
"google.golang.org/protobuf/proto"
|
|
|
|
pref "google.golang.org/protobuf/reflect/protoreflect"
|
|
|
|
"google.golang.org/protobuf/runtime/protoimpl"
|
2020-04-21 00:08:19 +00:00
|
|
|
"google.golang.org/protobuf/testing/protocmp"
|
2019-08-02 23:58:08 +00:00
|
|
|
|
2020-01-11 08:25:01 +00:00
|
|
|
legacy1pb "google.golang.org/protobuf/internal/testprotos/legacy/proto2_20160225_2fc053c5"
|
2019-08-02 23:58:08 +00:00
|
|
|
testpb "google.golang.org/protobuf/internal/testprotos/test"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestExtensionFuncs(t *testing.T) {
|
|
|
|
for _, test := range []struct {
|
|
|
|
message proto.Message
|
|
|
|
ext pref.ExtensionType
|
|
|
|
wantDefault interface{}
|
|
|
|
value interface{}
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
message: &testpb.TestAllExtensions{},
|
2020-02-02 08:53:34 +00:00
|
|
|
ext: testpb.E_OptionalInt32,
|
2019-08-02 23:58:08 +00:00
|
|
|
wantDefault: int32(0),
|
|
|
|
value: int32(1),
|
|
|
|
},
|
|
|
|
{
|
2019-10-05 01:57:48 +00:00
|
|
|
message: &testpb.TestAllExtensions{},
|
2020-02-02 08:53:34 +00:00
|
|
|
ext: testpb.E_RepeatedString,
|
2019-08-29 18:42:57 +00:00
|
|
|
wantDefault: ([]string)(nil),
|
|
|
|
value: []string{"a", "b", "c"},
|
2019-08-02 23:58:08 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
message: protoimpl.X.MessageOf(&legacy1pb.Message{}).Interface(),
|
|
|
|
ext: legacy1pb.E_Message_ExtensionOptionalBool,
|
|
|
|
wantDefault: false,
|
|
|
|
value: true,
|
|
|
|
},
|
|
|
|
} {
|
2019-08-28 18:08:22 +00:00
|
|
|
desc := fmt.Sprintf("Extension %v, value %v", test.ext.TypeDescriptor().FullName(), test.value)
|
2019-08-02 23:58:08 +00:00
|
|
|
if proto.HasExtension(test.message, test.ext) {
|
|
|
|
t.Errorf("%v:\nbefore setting extension HasExtension(...) = true, want false", desc)
|
|
|
|
}
|
|
|
|
got := proto.GetExtension(test.message, test.ext)
|
|
|
|
if d := cmp.Diff(test.wantDefault, got); d != "" {
|
|
|
|
t.Errorf("%v:\nbefore setting extension GetExtension(...) returns unexpected value (-want,+got):\n%v", desc, d)
|
|
|
|
}
|
|
|
|
proto.SetExtension(test.message, test.ext, test.value)
|
|
|
|
if !proto.HasExtension(test.message, test.ext) {
|
|
|
|
t.Errorf("%v:\nafter setting extension HasExtension(...) = false, want true", desc)
|
|
|
|
}
|
|
|
|
got = proto.GetExtension(test.message, test.ext)
|
|
|
|
if d := cmp.Diff(test.value, got); d != "" {
|
|
|
|
t.Errorf("%v:\nafter setting extension GetExtension(...) returns unexpected value (-want,+got):\n%v", desc, d)
|
|
|
|
}
|
|
|
|
proto.ClearExtension(test.message, test.ext)
|
|
|
|
if proto.HasExtension(test.message, test.ext) {
|
|
|
|
t.Errorf("%v:\nafter clearing extension HasExtension(...) = true, want false", desc)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-01-29 22:40:11 +00:00
|
|
|
|
2020-04-21 00:08:19 +00:00
|
|
|
func TestExtensionRanger(t *testing.T) {
|
|
|
|
want := map[pref.ExtensionType]interface{}{
|
|
|
|
testpb.E_OptionalInt32: int32(5),
|
|
|
|
testpb.E_OptionalString: string("hello"),
|
|
|
|
testpb.E_OptionalNestedMessage: &testpb.TestAllExtensions_NestedMessage{},
|
|
|
|
testpb.E_OptionalNestedEnum: testpb.TestAllTypes_BAZ,
|
|
|
|
testpb.E_RepeatedFloat: []float32{+32.32, -32.32},
|
|
|
|
testpb.E_RepeatedNestedMessage: []*testpb.TestAllExtensions_NestedMessage{{}},
|
|
|
|
testpb.E_RepeatedNestedEnum: []testpb.TestAllTypes_NestedEnum{testpb.TestAllTypes_BAZ},
|
|
|
|
}
|
|
|
|
|
|
|
|
m := &testpb.TestAllExtensions{}
|
|
|
|
for xt, v := range want {
|
|
|
|
proto.SetExtension(m, xt, v)
|
|
|
|
}
|
|
|
|
|
|
|
|
got := make(map[pref.ExtensionType]interface{})
|
|
|
|
proto.RangeExtensions(m, func(xt pref.ExtensionType, v interface{}) bool {
|
|
|
|
got[xt] = v
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
|
|
|
|
if diff := cmp.Diff(want, got, protocmp.Transform()); diff != "" {
|
|
|
|
t.Errorf("proto.RangeExtensions mismatch (-want +got):\n%s", diff)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-29 22:40:11 +00:00
|
|
|
func TestExtensionGetRace(t *testing.T) {
|
|
|
|
// Concurrently fetch an extension value while marshaling the message containing it.
|
|
|
|
// Create the message with proto.Unmarshal to give lazy extension decoding (if present)
|
|
|
|
// a chance to occur.
|
|
|
|
want := int32(42)
|
|
|
|
m1 := &testpb.TestAllExtensions{}
|
2020-02-02 08:53:34 +00:00
|
|
|
proto.SetExtension(m1, testpb.E_OptionalNestedMessage, &testpb.TestAllExtensions_NestedMessage{A: proto.Int32(want)})
|
2020-01-29 22:40:11 +00:00
|
|
|
b, err := proto.Marshal(m1)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
m := &testpb.TestAllExtensions{}
|
|
|
|
if err := proto.Unmarshal(b, m); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
for i := 0; i < 3; i++ {
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
if _, err := proto.Marshal(m); err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
2020-02-02 08:53:34 +00:00
|
|
|
got := proto.GetExtension(m, testpb.E_OptionalNestedMessage).(*testpb.TestAllExtensions_NestedMessage).GetA()
|
2020-01-29 22:40:11 +00:00
|
|
|
if got != want {
|
|
|
|
t.Errorf("GetExtension(optional_nested_message).a = %v, want %v", got, want)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
wg.Wait()
|
|
|
|
}
|