/* * * Copyright 2016 gRPC authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * */ package test_test import ( "context" "fmt" "net" "reflect" "sort" "testing" "time" "google.golang.org/grpc" "google.golang.org/grpc/credentials/insecure" "google.golang.org/grpc/internal/grpctest" "google.golang.org/grpc/reflection" "google.golang.org/grpc/reflection/internal" "google.golang.org/protobuf/proto" "google.golang.org/protobuf/reflect/protodesc" "google.golang.org/protobuf/reflect/protoreflect" "google.golang.org/protobuf/reflect/protoregistry" "google.golang.org/protobuf/types/descriptorpb" "google.golang.org/protobuf/types/dynamicpb" v1reflectiongrpc "google.golang.org/grpc/reflection/grpc_reflection_v1" v1reflectionpb "google.golang.org/grpc/reflection/grpc_reflection_v1" v1alphareflectiongrpc "google.golang.org/grpc/reflection/grpc_reflection_v1alpha" pb "google.golang.org/grpc/reflection/grpc_testing" pbv3 "google.golang.org/grpc/testdata/grpc_testing_not_regenerated" ) var ( s = reflection.NewServerV1(reflection.ServerOptions{}).(*internal.ServerReflectionServer) // fileDescriptor of each test proto file. fdProto2Ext *descriptorpb.FileDescriptorProto fdProto2Ext2 *descriptorpb.FileDescriptorProto fdDynamic *descriptorpb.FileDescriptorProto // reflection descriptors. fdDynamicFile protoreflect.FileDescriptor // fileDescriptor marshalled. fdTestByte []byte fdTestv3Byte []byte fdProto2Byte []byte fdProto2ExtByte []byte fdProto2Ext2Byte []byte fdDynamicByte []byte ) const defaultTestTimeout = 10 * time.Second type x struct { grpctest.Tester } func Test(t *testing.T) { grpctest.RunSubTests(t, x{}) } func loadFileDesc(filename string) (*descriptorpb.FileDescriptorProto, []byte) { fd, err := protoregistry.GlobalFiles.FindFileByPath(filename) if err != nil { panic(err) } fdProto := protodesc.ToFileDescriptorProto(fd) b, err := proto.Marshal(fdProto) if err != nil { panic(fmt.Sprintf("failed to marshal fd: %v", err)) } return fdProto, b } func loadFileDescDynamic(b []byte) (*descriptorpb.FileDescriptorProto, protoreflect.FileDescriptor, []byte) { m := new(descriptorpb.FileDescriptorProto) if err := proto.Unmarshal(b, m); err != nil { panic("failed to unmarshal dynamic proto raw descriptor") } fd, err := protodesc.NewFile(m, nil) if err != nil { panic(err) } err = protoregistry.GlobalFiles.RegisterFile(fd) if err != nil { panic(err) } for i := 0; i < fd.Messages().Len(); i++ { m := fd.Messages().Get(i) if err := protoregistry.GlobalTypes.RegisterMessage(dynamicpb.NewMessageType(m)); err != nil { panic(err) } } return m, fd, b } func init() { _, fdTestByte = loadFileDesc("reflection/grpc_testing/test.proto") _, fdTestv3Byte = loadFileDesc("testv3.proto") _, fdProto2Byte = loadFileDesc("reflection/grpc_testing/proto2.proto") fdProto2Ext, fdProto2ExtByte = loadFileDesc("reflection/grpc_testing/proto2_ext.proto") fdProto2Ext2, fdProto2Ext2Byte = loadFileDesc("reflection/grpc_testing/proto2_ext2.proto") fdDynamic, fdDynamicFile, fdDynamicByte = loadFileDescDynamic(pbv3.FileDynamicProtoRawDesc) } func (x) TestFileDescContainingExtension(t *testing.T) { for _, test := range []struct { st string extNum int32 want *descriptorpb.FileDescriptorProto }{ {"grpc.testing.ToBeExtended", 13, fdProto2Ext}, {"grpc.testing.ToBeExtended", 17, fdProto2Ext}, {"grpc.testing.ToBeExtended", 19, fdProto2Ext}, {"grpc.testing.ToBeExtended", 23, fdProto2Ext2}, {"grpc.testing.ToBeExtended", 29, fdProto2Ext2}, } { fd, err := s.FileDescEncodingContainingExtension(test.st, test.extNum, map[string]bool{}) if err != nil { t.Errorf("fileDescContainingExtension(%q) return error: %v", test.st, err) continue } var actualFd descriptorpb.FileDescriptorProto if err := proto.Unmarshal(fd[0], &actualFd); err != nil { t.Errorf("fileDescContainingExtension(%q) return invalid bytes: %v", test.st, err) continue } if !proto.Equal(&actualFd, test.want) { t.Errorf("fileDescContainingExtension(%q) returned %q, but wanted %q", test.st, &actualFd, test.want) } } } // intArray is used to sort []int32 type intArray []int32 func (s intArray) Len() int { return len(s) } func (s intArray) Swap(i, j int) { s[i], s[j] = s[j], s[i] } func (s intArray) Less(i, j int) bool { return s[i] < s[j] } func (x) TestAllExtensionNumbersForTypeName(t *testing.T) { for _, test := range []struct { st string want []int32 }{ {"grpc.testing.ToBeExtended", []int32{13, 17, 19, 23, 29}}, } { r, err := s.AllExtensionNumbersForTypeName(test.st) sort.Sort(intArray(r)) if err != nil || !reflect.DeepEqual(r, test.want) { t.Errorf("allExtensionNumbersForType(%q) = %v, %v, want %v, ", test.st, r, err, test.want) } } } func (x) TestFileDescWithDependencies(t *testing.T) { depFile, err := protodesc.NewFile( &descriptorpb.FileDescriptorProto{ Name: proto.String("dep.proto"), }, nil, ) if err != nil { t.Fatalf("unexpected error: %s", err) } deps := &protoregistry.Files{} if err := deps.RegisterFile(depFile); err != nil { t.Fatalf("unexpected error: %s", err) } rootFileProto := &descriptorpb.FileDescriptorProto{ Name: proto.String("root.proto"), Dependency: []string{ "google/protobuf/descriptor.proto", "reflection/grpc_testing/proto2_ext2.proto", "dep.proto", }, } // dep.proto is in deps; the other imports come from protoregistry.GlobalFiles resolver := &combinedResolver{first: protoregistry.GlobalFiles, second: deps} rootFile, err := protodesc.NewFile(rootFileProto, resolver) if err != nil { t.Fatalf("unexpected error: %s", err) } // Create a file hierarchy that contains a placeholder for dep.proto placeholderDep := placeholderFile{depFile} placeholderDeps := &protoregistry.Files{} if err := placeholderDeps.RegisterFile(placeholderDep); err != nil { t.Fatalf("unexpected error: %s", err) } resolver = &combinedResolver{first: protoregistry.GlobalFiles, second: placeholderDeps} rootFileHasPlaceholderDep, err := protodesc.NewFile(rootFileProto, resolver) if err != nil { t.Fatalf("unexpected error: %s", err) } rootFileIsPlaceholder := placeholderFile{rootFile} // Full transitive dependency graph of root.proto includes five files: // - root.proto // - google/protobuf/descriptor.proto // - reflection/grpc_testing/proto2_ext2.proto // - reflection/grpc_testing/proto2.proto // - dep.proto for _, test := range []struct { name string sent []string root protoreflect.FileDescriptor expect []string }{ { name: "send_all", root: rootFile, // expect full transitive closure expect: []string{ "root.proto", "google/protobuf/descriptor.proto", "reflection/grpc_testing/proto2_ext2.proto", "reflection/grpc_testing/proto2.proto", "dep.proto", }, }, { name: "already_sent", sent: []string{ "root.proto", "google/protobuf/descriptor.proto", "reflection/grpc_testing/proto2_ext2.proto", "reflection/grpc_testing/proto2.proto", "dep.proto", }, root: rootFile, // expect only the root to be re-sent expect: []string{"root.proto"}, }, { name: "some_already_sent", sent: []string{ "reflection/grpc_testing/proto2_ext2.proto", "reflection/grpc_testing/proto2.proto", }, root: rootFile, expect: []string{ "root.proto", "google/protobuf/descriptor.proto", "dep.proto", }, }, { name: "root_is_placeholder", root: rootFileIsPlaceholder, // expect error, no files }, { name: "placeholder_skipped", root: rootFileHasPlaceholderDep, // dep.proto is a placeholder so is skipped expect: []string{ "root.proto", "google/protobuf/descriptor.proto", "reflection/grpc_testing/proto2_ext2.proto", "reflection/grpc_testing/proto2.proto", }, }, { name: "placeholder_skipped_and_some_sent", sent: []string{ "reflection/grpc_testing/proto2_ext2.proto", "reflection/grpc_testing/proto2.proto", }, root: rootFileHasPlaceholderDep, expect: []string{ "root.proto", "google/protobuf/descriptor.proto", }, }, } { t.Run(test.name, func(t *testing.T) { s := reflection.NewServerV1(reflection.ServerOptions{}).(*internal.ServerReflectionServer) sent := map[string]bool{} for _, path := range test.sent { sent[path] = true } descriptors, err := s.FileDescWithDependencies(test.root, sent) if len(test.expect) == 0 { // if we're not expecting any files then we're expecting an error if err == nil { t.Fatalf("expecting an error; instead got %d files", len(descriptors)) } return } checkDescriptorResults(t, descriptors, test.expect) }) } } func checkDescriptorResults(t *testing.T, descriptors [][]byte, expect []string) { t.Helper() if len(descriptors) != len(expect) { t.Errorf("expected result to contain %d descriptor(s); instead got %d", len(expect), len(descriptors)) } names := map[string]struct{}{} for i, desc := range descriptors { var descProto descriptorpb.FileDescriptorProto if err := proto.Unmarshal(desc, &descProto); err != nil { t.Fatalf("could not unmarshal descriptor result #%d", i+1) } names[descProto.GetName()] = struct{}{} } actual := make([]string, 0, len(names)) for name := range names { actual = append(actual, name) } sort.Strings(actual) sort.Strings(expect) if !reflect.DeepEqual(actual, expect) { t.Fatalf("expected file descriptors for %v; instead got %v", expect, actual) } } type placeholderFile struct { protoreflect.FileDescriptor } func (placeholderFile) IsPlaceholder() bool { return true } type combinedResolver struct { first, second protodesc.Resolver } func (r *combinedResolver) FindFileByPath(path string) (protoreflect.FileDescriptor, error) { file, err := r.first.FindFileByPath(path) if err == nil { return file, nil } return r.second.FindFileByPath(path) } func (r *combinedResolver) FindDescriptorByName(name protoreflect.FullName) (protoreflect.Descriptor, error) { desc, err := r.first.FindDescriptorByName(name) if err == nil { return desc, nil } return r.second.FindDescriptorByName(name) } // Do end2end tests. type server struct { pb.UnimplementedSearchServiceServer } func (s *server) Search(ctx context.Context, in *pb.SearchRequest) (*pb.SearchResponse, error) { return &pb.SearchResponse{}, nil } func (s *server) StreamingSearch(stream pb.SearchService_StreamingSearchServer) error { return nil } type serverV3 struct{} func (s *serverV3) Search(ctx context.Context, in *pbv3.SearchRequestV3) (*pbv3.SearchResponseV3, error) { return &pbv3.SearchResponseV3{}, nil } func (s *serverV3) StreamingSearch(stream pbv3.SearchServiceV3_StreamingSearchServer) error { return nil } func (x) TestReflectionEnd2end(t *testing.T) { // Start server. lis, err := net.Listen("tcp", "localhost:0") if err != nil { t.Fatalf("failed to listen: %v", err) } s := grpc.NewServer() pb.RegisterSearchServiceServer(s, &server{}) pbv3.RegisterSearchServiceV3Server(s, &serverV3{}) registerDynamicProto(s, fdDynamic, fdDynamicFile) // Register reflection service on s. reflection.Register(s) go s.Serve(lis) t.Cleanup(s.Stop) // Create client. conn, err := grpc.NewClient(lis.Addr().String(), grpc.WithTransportCredentials(insecure.NewCredentials())) if err != nil { t.Fatalf("cannot connect to server: %v", err) } defer conn.Close() clientV1 := v1reflectiongrpc.NewServerReflectionClient(conn) clientV1Alpha := v1alphareflectiongrpc.NewServerReflectionClient(conn) testCases := []struct { name string client v1reflectiongrpc.ServerReflectionClient }{ { name: "v1", client: clientV1, }, { name: "v1alpha", client: v1AlphaClientAdapter{stub: clientV1Alpha}, }, } for _, testCase := range testCases { c := testCase.client t.Run(testCase.name, func(t *testing.T) { ctx, cancel := context.WithTimeout(context.Background(), defaultTestTimeout) defer cancel() stream, err := c.ServerReflectionInfo(ctx, grpc.WaitForReady(true)) if err != nil { t.Fatalf("cannot get ServerReflectionInfo: %v", err) } testFileByFilenameTransitiveClosure(t, stream, true) testFileByFilenameTransitiveClosure(t, stream, false) testFileByFilename(t, stream) testFileByFilenameError(t, stream) testFileContainingSymbol(t, stream) testFileContainingSymbolError(t, stream) testFileContainingExtension(t, stream) testFileContainingExtensionError(t, stream) testAllExtensionNumbersOfType(t, stream) testAllExtensionNumbersOfTypeError(t, stream) testListServices(t, stream) }) } } func testFileByFilenameTransitiveClosure(t *testing.T, stream v1reflectiongrpc.ServerReflection_ServerReflectionInfoClient, expectClosure bool) { filename := "reflection/grpc_testing/proto2_ext2.proto" if err := stream.Send(&v1reflectionpb.ServerReflectionRequest{ MessageRequest: &v1reflectionpb.ServerReflectionRequest_FileByFilename{ FileByFilename: filename, }, }); err != nil { t.Fatalf("failed to send request: %v", err) } r, err := stream.Recv() if err != nil { // io.EOF is not ok. t.Fatalf("failed to recv response: %v", err) } switch r.MessageResponse.(type) { case *v1reflectionpb.ServerReflectionResponse_FileDescriptorResponse: if !reflect.DeepEqual(r.GetFileDescriptorResponse().FileDescriptorProto[0], fdProto2Ext2Byte) { t.Errorf("FileByFilename(%v)\nreceived: %q,\nwant: %q", filename, r.GetFileDescriptorResponse().FileDescriptorProto[0], fdProto2Ext2Byte) } if expectClosure { if len(r.GetFileDescriptorResponse().FileDescriptorProto) != 2 { t.Errorf("FileByFilename(%v) returned %v file descriptors, expected 2", filename, len(r.GetFileDescriptorResponse().FileDescriptorProto)) } else if !reflect.DeepEqual(r.GetFileDescriptorResponse().FileDescriptorProto[1], fdProto2Byte) { t.Errorf("FileByFilename(%v)\nreceived: %q,\nwant: %q", filename, r.GetFileDescriptorResponse().FileDescriptorProto[1], fdProto2Byte) } } else if len(r.GetFileDescriptorResponse().FileDescriptorProto) != 1 { t.Errorf("FileByFilename(%v) returned %v file descriptors, expected 1", filename, len(r.GetFileDescriptorResponse().FileDescriptorProto)) } default: t.Errorf("FileByFilename(%v) = %v, want type ", filename, r.MessageResponse) } } func testFileByFilename(t *testing.T, stream v1reflectiongrpc.ServerReflection_ServerReflectionInfoClient) { for _, test := range []struct { filename string want []byte }{ {"reflection/grpc_testing/test.proto", fdTestByte}, {"reflection/grpc_testing/proto2.proto", fdProto2Byte}, {"reflection/grpc_testing/proto2_ext.proto", fdProto2ExtByte}, {"dynamic.proto", fdDynamicByte}, } { if err := stream.Send(&v1reflectionpb.ServerReflectionRequest{ MessageRequest: &v1reflectionpb.ServerReflectionRequest_FileByFilename{ FileByFilename: test.filename, }, }); err != nil { t.Fatalf("failed to send request: %v", err) } r, err := stream.Recv() if err != nil { // io.EOF is not ok. t.Fatalf("failed to recv response: %v", err) } switch r.MessageResponse.(type) { case *v1reflectionpb.ServerReflectionResponse_FileDescriptorResponse: if !reflect.DeepEqual(r.GetFileDescriptorResponse().FileDescriptorProto[0], test.want) { t.Errorf("FileByFilename(%v)\nreceived: %q,\nwant: %q", test.filename, r.GetFileDescriptorResponse().FileDescriptorProto[0], test.want) } default: t.Errorf("FileByFilename(%v) = %v, want type ", test.filename, r.MessageResponse) } } } func testFileByFilenameError(t *testing.T, stream v1reflectiongrpc.ServerReflection_ServerReflectionInfoClient) { for _, test := range []string{ "test.poto", "proo2.proto", "proto2_et.proto", } { if err := stream.Send(&v1reflectionpb.ServerReflectionRequest{ MessageRequest: &v1reflectionpb.ServerReflectionRequest_FileByFilename{ FileByFilename: test, }, }); err != nil { t.Fatalf("failed to send request: %v", err) } r, err := stream.Recv() if err != nil { // io.EOF is not ok. t.Fatalf("failed to recv response: %v", err) } switch r.MessageResponse.(type) { case *v1reflectionpb.ServerReflectionResponse_ErrorResponse: default: t.Errorf("FileByFilename(%v) = %v, want type ", test, r.MessageResponse) } } } func testFileContainingSymbol(t *testing.T, stream v1reflectiongrpc.ServerReflection_ServerReflectionInfoClient) { for _, test := range []struct { symbol string want []byte }{ {"grpc.testing.SearchService", fdTestByte}, {"grpc.testing.SearchService.Search", fdTestByte}, {"grpc.testing.SearchService.StreamingSearch", fdTestByte}, {"grpc.testing.SearchResponse", fdTestByte}, {"grpc.testing.ToBeExtended", fdProto2Byte}, // Test support package v3. {"grpc.testingv3.SearchServiceV3", fdTestv3Byte}, {"grpc.testingv3.SearchServiceV3.Search", fdTestv3Byte}, {"grpc.testingv3.SearchServiceV3.StreamingSearch", fdTestv3Byte}, {"grpc.testingv3.SearchResponseV3", fdTestv3Byte}, // search for field, oneof, enum, and enum value symbols, too {"grpc.testingv3.SearchResponseV3.Result.snippets", fdTestv3Byte}, {"grpc.testingv3.SearchResponseV3.Result.Value.val", fdTestv3Byte}, {"grpc.testingv3.SearchResponseV3.Result.Value.str", fdTestv3Byte}, {"grpc.testingv3.SearchResponseV3.State", fdTestv3Byte}, {"grpc.testingv3.SearchResponseV3.FRESH", fdTestv3Byte}, // Test dynamic symbols {"grpc.testing.DynamicService", fdDynamicByte}, {"grpc.testing.DynamicReq", fdDynamicByte}, {"grpc.testing.DynamicRes", fdDynamicByte}, } { if err := stream.Send(&v1reflectionpb.ServerReflectionRequest{ MessageRequest: &v1reflectionpb.ServerReflectionRequest_FileContainingSymbol{ FileContainingSymbol: test.symbol, }, }); err != nil { t.Fatalf("failed to send request: %v", err) } r, err := stream.Recv() if err != nil { // io.EOF is not ok. t.Fatalf("failed to recv response: %v", err) } switch r.MessageResponse.(type) { case *v1reflectionpb.ServerReflectionResponse_FileDescriptorResponse: if !reflect.DeepEqual(r.GetFileDescriptorResponse().FileDescriptorProto[0], test.want) { t.Errorf("FileContainingSymbol(%v)\nreceived: %q,\nwant: %q", test.symbol, r.GetFileDescriptorResponse().FileDescriptorProto[0], test.want) } default: t.Errorf("FileContainingSymbol(%v) = %v, want type ", test.symbol, r.MessageResponse) } } } func testFileContainingSymbolError(t *testing.T, stream v1reflectiongrpc.ServerReflection_ServerReflectionInfoClient) { for _, test := range []string{ "grpc.testing.SearchService_", "grpc.testing.SearchService.SearchE", "grpc.testing_.SearchResponse", "gpc.testing.ToBeExtended", } { if err := stream.Send(&v1reflectionpb.ServerReflectionRequest{ MessageRequest: &v1reflectionpb.ServerReflectionRequest_FileContainingSymbol{ FileContainingSymbol: test, }, }); err != nil { t.Fatalf("failed to send request: %v", err) } r, err := stream.Recv() if err != nil { // io.EOF is not ok. t.Fatalf("failed to recv response: %v", err) } switch r.MessageResponse.(type) { case *v1reflectionpb.ServerReflectionResponse_ErrorResponse: default: t.Errorf("FileContainingSymbol(%v) = %v, want type ", test, r.MessageResponse) } } } func testFileContainingExtension(t *testing.T, stream v1reflectiongrpc.ServerReflection_ServerReflectionInfoClient) { for _, test := range []struct { typeName string extNum int32 want []byte }{ {"grpc.testing.ToBeExtended", 13, fdProto2ExtByte}, {"grpc.testing.ToBeExtended", 17, fdProto2ExtByte}, {"grpc.testing.ToBeExtended", 19, fdProto2ExtByte}, {"grpc.testing.ToBeExtended", 23, fdProto2Ext2Byte}, {"grpc.testing.ToBeExtended", 29, fdProto2Ext2Byte}, } { if err := stream.Send(&v1reflectionpb.ServerReflectionRequest{ MessageRequest: &v1reflectionpb.ServerReflectionRequest_FileContainingExtension{ FileContainingExtension: &v1reflectionpb.ExtensionRequest{ ContainingType: test.typeName, ExtensionNumber: test.extNum, }, }, }); err != nil { t.Fatalf("failed to send request: %v", err) } r, err := stream.Recv() if err != nil { // io.EOF is not ok. t.Fatalf("failed to recv response: %v", err) } switch r.MessageResponse.(type) { case *v1reflectionpb.ServerReflectionResponse_FileDescriptorResponse: if !reflect.DeepEqual(r.GetFileDescriptorResponse().FileDescriptorProto[0], test.want) { t.Errorf("FileContainingExtension(%v, %v)\nreceived: %q,\nwant: %q", test.typeName, test.extNum, r.GetFileDescriptorResponse().FileDescriptorProto[0], test.want) } default: t.Errorf("FileContainingExtension(%v, %v) = %v, want type ", test.typeName, test.extNum, r.MessageResponse) } } } func testFileContainingExtensionError(t *testing.T, stream v1reflectiongrpc.ServerReflection_ServerReflectionInfoClient) { for _, test := range []struct { typeName string extNum int32 }{ {"grpc.testing.ToBExtended", 17}, {"grpc.testing.ToBeExtended", 15}, } { if err := stream.Send(&v1reflectionpb.ServerReflectionRequest{ MessageRequest: &v1reflectionpb.ServerReflectionRequest_FileContainingExtension{ FileContainingExtension: &v1reflectionpb.ExtensionRequest{ ContainingType: test.typeName, ExtensionNumber: test.extNum, }, }, }); err != nil { t.Fatalf("failed to send request: %v", err) } r, err := stream.Recv() if err != nil { // io.EOF is not ok. t.Fatalf("failed to recv response: %v", err) } switch r.MessageResponse.(type) { case *v1reflectionpb.ServerReflectionResponse_ErrorResponse: default: t.Errorf("FileContainingExtension(%v, %v) = %v, want type ", test.typeName, test.extNum, r.MessageResponse) } } } func testAllExtensionNumbersOfType(t *testing.T, stream v1reflectiongrpc.ServerReflection_ServerReflectionInfoClient) { for _, test := range []struct { typeName string want []int32 }{ {"grpc.testing.ToBeExtended", []int32{13, 17, 19, 23, 29}}, {"grpc.testing.DynamicReq", nil}, } { if err := stream.Send(&v1reflectionpb.ServerReflectionRequest{ MessageRequest: &v1reflectionpb.ServerReflectionRequest_AllExtensionNumbersOfType{ AllExtensionNumbersOfType: test.typeName, }, }); err != nil { t.Fatalf("failed to send request: %v", err) } r, err := stream.Recv() if err != nil { // io.EOF is not ok. t.Fatalf("failed to recv response: %v", err) } switch r.MessageResponse.(type) { case *v1reflectionpb.ServerReflectionResponse_AllExtensionNumbersResponse: extNum := r.GetAllExtensionNumbersResponse().ExtensionNumber sort.Sort(intArray(extNum)) if r.GetAllExtensionNumbersResponse().BaseTypeName != test.typeName || !reflect.DeepEqual(extNum, test.want) { t.Errorf("AllExtensionNumbersOfType(%v)\nreceived: %v,\nwant: {%q %v}", r.GetAllExtensionNumbersResponse(), test.typeName, test.typeName, test.want) } default: t.Errorf("AllExtensionNumbersOfType(%v) = %v, want type ", test.typeName, r.MessageResponse) } } } func testAllExtensionNumbersOfTypeError(t *testing.T, stream v1reflectiongrpc.ServerReflection_ServerReflectionInfoClient) { for _, test := range []string{ "grpc.testing.ToBeExtendedE", } { if err := stream.Send(&v1reflectionpb.ServerReflectionRequest{ MessageRequest: &v1reflectionpb.ServerReflectionRequest_AllExtensionNumbersOfType{ AllExtensionNumbersOfType: test, }, }); err != nil { t.Fatalf("failed to send request: %v", err) } r, err := stream.Recv() if err != nil { // io.EOF is not ok. t.Fatalf("failed to recv response: %v", err) } switch r.MessageResponse.(type) { case *v1reflectionpb.ServerReflectionResponse_ErrorResponse: default: t.Errorf("AllExtensionNumbersOfType(%v) = %v, want type ", test, r.MessageResponse) } } } func testListServices(t *testing.T, stream v1reflectiongrpc.ServerReflection_ServerReflectionInfoClient) { if err := stream.Send(&v1reflectionpb.ServerReflectionRequest{ MessageRequest: &v1reflectionpb.ServerReflectionRequest_ListServices{}, }); err != nil { t.Fatalf("failed to send request: %v", err) } r, err := stream.Recv() if err != nil { // io.EOF is not ok. t.Fatalf("failed to recv response: %v", err) } switch r.MessageResponse.(type) { case *v1reflectionpb.ServerReflectionResponse_ListServicesResponse: services := r.GetListServicesResponse().Service want := []string{ "grpc.testingv3.SearchServiceV3", "grpc.testing.SearchService", "grpc.reflection.v1.ServerReflection", "grpc.reflection.v1alpha.ServerReflection", "grpc.testing.DynamicService", } // Compare service names in response with want. if len(services) != len(want) { t.Errorf("= %v, want service names: %v", services, want) } m := make(map[string]int) for _, e := range services { m[e.Name]++ } for _, e := range want { if m[e] > 0 { m[e]-- continue } t.Errorf("ListService\nreceived: %v,\nwant: %q", services, want) } default: t.Errorf("ListServices = %v, want type ", r.MessageResponse) } } func registerDynamicProto(srv *grpc.Server, fdp *descriptorpb.FileDescriptorProto, fd protoreflect.FileDescriptor) { type emptyInterface any for i := 0; i < fd.Services().Len(); i++ { s := fd.Services().Get(i) sd := &grpc.ServiceDesc{ ServiceName: string(s.FullName()), HandlerType: (*emptyInterface)(nil), Metadata: fdp.GetName(), } for j := 0; j < s.Methods().Len(); j++ { m := s.Methods().Get(j) sd.Methods = append(sd.Methods, grpc.MethodDesc{ MethodName: string(m.Name()), }) } srv.RegisterService(sd, struct{}{}) } } type v1AlphaClientAdapter struct { stub v1alphareflectiongrpc.ServerReflectionClient } func (v v1AlphaClientAdapter) ServerReflectionInfo(ctx context.Context, opts ...grpc.CallOption) (v1reflectiongrpc.ServerReflection_ServerReflectionInfoClient, error) { stream, err := v.stub.ServerReflectionInfo(ctx, opts...) if err != nil { return nil, err } return v1AlphaClientStreamAdapter{stream}, nil } type v1AlphaClientStreamAdapter struct { v1alphareflectiongrpc.ServerReflection_ServerReflectionInfoClient } func (s v1AlphaClientStreamAdapter) Send(request *v1reflectionpb.ServerReflectionRequest) error { return s.ServerReflection_ServerReflectionInfoClient.Send(internal.V1ToV1AlphaRequest(request)) } func (s v1AlphaClientStreamAdapter) Recv() (*v1reflectionpb.ServerReflectionResponse, error) { resp, err := s.ServerReflection_ServerReflectionInfoClient.Recv() if err != nil { return nil, err } return internal.V1AlphaToV1Response(resp), nil }