mirror of
https://github.com/protocolbuffers/protobuf-go.git
synced 2025-01-04 02:38:50 +00:00
b6405bd782
Implement support in protoc-gen-go for generating messages and enums that satisfy the v2 protobuf reflection interfaces. Specifically, the following are added: * top-level variable representing the file descriptor * ProtoReflect method on enums (to implement protoV2.Enum) * ProtoReflect method on messages (to implement protoV2.Message) The following are not supported yet: * resolving transitive dependencies for file imports * Extension descriptors * Service descriptors The implementation approach creates a single array for all the message and enum declarations and references sections of that array to complete dependencies. Since protobuf declarations can form a graph (a message may depend on itself), it is difficult to construct a graph as a single literal. One way is to use placeholder descriptors, but that is not efficient as it requires encoding the full name of each dependent enum and message and then later resolving it; thus, both expanding the binary size and also increasing initialization cost. Instead, we add a prototype.{Enum,Message}.Reference method to obtain a descriptor reference for the purposes for satisfying dependencies. As such, nested declarations and dependencies are populated in an init function. Other changes to support the implementation: * Added a protoimpl package to expose the MessageType type and also the MessageTypeOf and EnumTypeOf helper functions. * Added a protogen.File.GoIdent field to provide a suggested variable name for the file descriptor. * Added prototype.{Enum,Message}.Reference that provides a descriptor reference for the purposes for satisfying cyclic dependencies. * Added protoreflect.{Syntax,Cardinality,Kind}.GoString to obtain a Go source identifier that represents the given constant. Change-Id: I9455764882dee6ad10f251901e7d419091e8bf1d Reviewed-on: https://go-review.googlesource.com/c/150074 Reviewed-by: Damien Neil <dneil@google.com>
61 lines
1.7 KiB
Go
61 lines
1.7 KiB
Go
// 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 protogen
|
|
|
|
import "testing"
|
|
|
|
func TestCamelCase(t *testing.T) {
|
|
tests := []struct {
|
|
in, want string
|
|
}{
|
|
{"one", "One"},
|
|
{"one_two", "OneTwo"},
|
|
{"_my_field_name_2", "XMyFieldName_2"},
|
|
{"Something_Capped", "Something_Capped"},
|
|
{"my_Name", "My_Name"},
|
|
{"OneTwo", "OneTwo"},
|
|
{"_", "X"},
|
|
{"_a_", "XA_"},
|
|
{"one.two", "OneTwo"},
|
|
{"one.Two", "One_Two"},
|
|
{"one_two.three_four", "OneTwoThreeFour"},
|
|
{"one_two.Three_four", "OneTwo_ThreeFour"},
|
|
{"_one._two", "XOne_XTwo"},
|
|
{"SCREAMING_SNAKE_CASE", "SCREAMING_SNAKE_CASE"},
|
|
{"double__underscore", "Double_Underscore"},
|
|
{"camelCase", "CamelCase"},
|
|
{"go2proto", "Go2Proto"},
|
|
{"世界", "世界"},
|
|
{"x世界", "X世界"},
|
|
{"foo_bar世界", "FooBar世界"},
|
|
}
|
|
for _, tc := range tests {
|
|
if got := camelCase(tc.in); got != tc.want {
|
|
t.Errorf("CamelCase(%q) = %q, want %q", tc.in, got, tc.want)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestCleanGoName(t *testing.T) {
|
|
tests := []struct {
|
|
in, want, wantExported string
|
|
}{
|
|
{"", "", "X"},
|
|
{"hello", "hello", "Hello"},
|
|
{"hello-world!!", "hello_world__", "Hello_world__"},
|
|
{"hello-\xde\xad\xbe\xef\x00", "hello_____", "Hello_____"},
|
|
{"hello 世界", "hello_世界", "Hello_世界"},
|
|
{"世界", "世界", "X世界"},
|
|
}
|
|
for _, tc := range tests {
|
|
if got := cleanGoName(tc.in, false); got != tc.want {
|
|
t.Errorf("cleanGoName(%q, false) = %q, want %q", tc.in, got, tc.want)
|
|
}
|
|
if got := cleanGoName(tc.in, true); got != tc.wantExported {
|
|
t.Errorf("cleanGoName(%q, true) = %q, want %q", tc.in, got, tc.wantExported)
|
|
}
|
|
}
|
|
}
|