From b3645818f0d0683e5f2bcb4a71db7e6319d6c2e4 Mon Sep 17 00:00:00 2001 From: katelyn martin Date: Wed, 15 May 2024 00:00:00 +0000 Subject: [PATCH] =?UTF-8?q?tonic-reflection:=20=F0=9F=86=95=20use=20`v1`?= =?UTF-8?q?=20reflection=20protobuffer=20definitions?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit this fixes #1685. in penumbra-zone/penumbra#4392, we observed that tonic servers do not properly support reflection when servicing a request sent by recent versions of [`grpcurl`], after [v1.8.8] began using `grpc.reflection.v1.ServerReflection`. (see fullstorydev/grpcurl#407) these lead to an error regarding an unexpected status code, like this: ``` ❯ grpcurl --version grpcurl v1.9.1 ❯ grpcurl -vv grpc.testnet.penumbra.zone:443 list Failed to list services: rpc error: code = Unknown desc = unexpected HTTP status code received from server: 405 (Method Not Allowed); malformed header: missing HTTP content-type ``` this adds the v1 reflection definition to `tonic-reflection`, which was observed as fixing these issues for our gRPC endpoint. ### 🩹 changes changes in this commet are as follows: * vendors the `v1` definition of [`reflection.proto`][proto]. * renames the (deprecated) `v1alpha` definition to `reflection_v1alpha.proto`. * `tonic_reflection::generated::grpc_reflection_v1` links to the generated Rust code (created by running `cargo run --package codegen`). * `tonic_reflection::generated::FILE_DESCRIPTOR_SET` is replaced by `tonic_reflection::generated::{FILE_DESCRIPTOR_SET_V1ALPHA, FILE_DESCRIPTOR_SET_V1}`. * `tonic_reflection::pb` now contains namespaced `tonic_reflection::pb::{v1alpha, v1}` submodules. (**NB: this is a breaking change to the public `tonic-reflection` API.**) * `tonic_reflection::server` is updated to use the generated `tonic_reflection::pb::v1` code. * `HttpsUriWithoutTlsSupport` is now gated behind the `transport` feature flag, because it is not used otherwise. [v1.8.8]: https://github.com/fullstorydev/grpcurl/releases/tag/v1.8.8 [proto]: https://github.com/grpc/grpc/blob/master/src/proto/grpc/reflection/v1/reflection.proto [grpcurl]: https://github.com/fullstorydev/grpcurl --- fixes: #1685 x-ref: penumbra-zone/penumbra#4392 co-authored-by: Conor Schaefer --- codegen/src/main.rs | 14 +- tonic-reflection/proto/reflection_v1.proto | 147 ++++++ ...lection.proto => reflection_v1alpha.proto} | 0 .../src/generated/grpc_reflection_v1.rs | 465 ++++++++++++++++++ .../src/generated/reflection_v1.bin | Bin 0 -> 7639 bytes .../src/generated/reflection_v1alpha1.bin | Bin 7190 -> 7198 bytes tonic-reflection/src/lib.rs | 37 +- tonic-reflection/src/server.rs | 11 +- tonic-reflection/tests/server.rs | 8 +- tonic-types/src/generated/types.bin | Bin 24204 -> 24305 bytes tonic/src/transport/service/connector.rs | 1 + 11 files changed, 667 insertions(+), 16 deletions(-) create mode 100644 tonic-reflection/proto/reflection_v1.proto rename tonic-reflection/proto/{reflection.proto => reflection_v1alpha.proto} (100%) create mode 100644 tonic-reflection/src/generated/grpc_reflection_v1.rs create mode 100644 tonic-reflection/src/generated/reflection_v1.bin diff --git a/codegen/src/main.rs b/codegen/src/main.rs index 7659ad93d..7eb2d102c 100644 --- a/codegen/src/main.rs +++ b/codegen/src/main.rs @@ -21,7 +21,19 @@ fn main() { .parent() .unwrap() .join("tonic-reflection"), - &["proto/reflection.proto"], + &["proto/reflection_v1.proto"], + &["proto"], + &PathBuf::from("src/generated"), + &PathBuf::from("src/generated/reflection_v1.bin"), + true, + true, + ); + codegen( + &PathBuf::from(std::env!("CARGO_MANIFEST_DIR")) + .parent() + .unwrap() + .join("tonic-reflection"), + &["proto/reflection_v1alpha.proto"], &["proto"], &PathBuf::from("src/generated"), &PathBuf::from("src/generated/reflection_v1alpha1.bin"), diff --git a/tonic-reflection/proto/reflection_v1.proto b/tonic-reflection/proto/reflection_v1.proto new file mode 100644 index 000000000..1a2ceedc3 --- /dev/null +++ b/tonic-reflection/proto/reflection_v1.proto @@ -0,0 +1,147 @@ +// Copyright 2016 The 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. + +// Service exported by server reflection. A more complete description of how +// server reflection works can be found at +// https://github.com/grpc/grpc/blob/master/doc/server-reflection.md +// +// The canonical version of this proto can be found at +// https://github.com/grpc/grpc-proto/blob/master/grpc/reflection/v1/reflection.proto + +syntax = "proto3"; + +package grpc.reflection.v1; + +option go_package = "google.golang.org/grpc/reflection/grpc_reflection_v1"; +option java_multiple_files = true; +option java_package = "io.grpc.reflection.v1"; +option java_outer_classname = "ServerReflectionProto"; + +service ServerReflection { + // The reflection service is structured as a bidirectional stream, ensuring + // all related requests go to a single server. + rpc ServerReflectionInfo(stream ServerReflectionRequest) + returns (stream ServerReflectionResponse); +} + +// The message sent by the client when calling ServerReflectionInfo method. +message ServerReflectionRequest { + string host = 1; + // To use reflection service, the client should set one of the following + // fields in message_request. The server distinguishes requests by their + // defined field and then handles them using corresponding methods. + oneof message_request { + // Find a proto file by the file name. + string file_by_filename = 3; + + // Find the proto file that declares the given fully-qualified symbol name. + // This field should be a fully-qualified symbol name + // (e.g. .[.] or .). + string file_containing_symbol = 4; + + // Find the proto file which defines an extension extending the given + // message type with the given field number. + ExtensionRequest file_containing_extension = 5; + + // Finds the tag numbers used by all known extensions of the given message + // type, and appends them to ExtensionNumberResponse in an undefined order. + // Its corresponding method is best-effort: it's not guaranteed that the + // reflection service will implement this method, and it's not guaranteed + // that this method will provide all extensions. Returns + // StatusCode::UNIMPLEMENTED if it's not implemented. + // This field should be a fully-qualified type name. The format is + // . + string all_extension_numbers_of_type = 6; + + // List the full names of registered services. The content will not be + // checked. + string list_services = 7; + } +} + +// The type name and extension number sent by the client when requesting +// file_containing_extension. +message ExtensionRequest { + // Fully-qualified type name. The format should be . + string containing_type = 1; + int32 extension_number = 2; +} + +// The message sent by the server to answer ServerReflectionInfo method. +message ServerReflectionResponse { + string valid_host = 1; + ServerReflectionRequest original_request = 2; + // The server sets one of the following fields according to the message_request + // in the request. + oneof message_response { + // This message is used to answer file_by_filename, file_containing_symbol, + // file_containing_extension requests with transitive dependencies. + // As the repeated label is not allowed in oneof fields, we use a + // FileDescriptorResponse message to encapsulate the repeated fields. + // The reflection service is allowed to avoid sending FileDescriptorProtos + // that were previously sent in response to earlier requests in the stream. + FileDescriptorResponse file_descriptor_response = 4; + + // This message is used to answer all_extension_numbers_of_type requests. + ExtensionNumberResponse all_extension_numbers_response = 5; + + // This message is used to answer list_services requests. + ListServiceResponse list_services_response = 6; + + // This message is used when an error occurs. + ErrorResponse error_response = 7; + } +} + +// Serialized FileDescriptorProto messages sent by the server answering +// a file_by_filename, file_containing_symbol, or file_containing_extension +// request. +message FileDescriptorResponse { + // Serialized FileDescriptorProto messages. We avoid taking a dependency on + // descriptor.proto, which uses proto2 only features, by making them opaque + // bytes instead. + repeated bytes file_descriptor_proto = 1; +} + +// A list of extension numbers sent by the server answering +// all_extension_numbers_of_type request. +message ExtensionNumberResponse { + // Full name of the base type, including the package name. The format + // is . + string base_type_name = 1; + repeated int32 extension_number = 2; +} + +// A list of ServiceResponse sent by the server answering list_services request. +message ListServiceResponse { + // The information of each service may be expanded in the future, so we use + // ServiceResponse message to encapsulate it. + repeated ServiceResponse service = 1; +} + +// The information of a single service used by ListServiceResponse to answer +// list_services request. +message ServiceResponse { + // Full name of a registered service, including its package name. The format + // is . + string name = 1; +} + +// The error code and error message sent by the server when an error occurs. +message ErrorResponse { + // This field uses the error codes defined in grpc::StatusCode. + int32 error_code = 1; + string error_message = 2; +} + diff --git a/tonic-reflection/proto/reflection.proto b/tonic-reflection/proto/reflection_v1alpha.proto similarity index 100% rename from tonic-reflection/proto/reflection.proto rename to tonic-reflection/proto/reflection_v1alpha.proto diff --git a/tonic-reflection/src/generated/grpc_reflection_v1.rs b/tonic-reflection/src/generated/grpc_reflection_v1.rs new file mode 100644 index 000000000..42154f16d --- /dev/null +++ b/tonic-reflection/src/generated/grpc_reflection_v1.rs @@ -0,0 +1,465 @@ +// This file is @generated by prost-build. +/// The message sent by the client when calling ServerReflectionInfo method. +#[allow(clippy::derive_partial_eq_without_eq)] +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ServerReflectionRequest { + #[prost(string, tag = "1")] + pub host: ::prost::alloc::string::String, + /// To use reflection service, the client should set one of the following + /// fields in message_request. The server distinguishes requests by their + /// defined field and then handles them using corresponding methods. + #[prost(oneof = "server_reflection_request::MessageRequest", tags = "3, 4, 5, 6, 7")] + pub message_request: ::core::option::Option< + server_reflection_request::MessageRequest, + >, +} +/// Nested message and enum types in `ServerReflectionRequest`. +pub mod server_reflection_request { + /// To use reflection service, the client should set one of the following + /// fields in message_request. The server distinguishes requests by their + /// defined field and then handles them using corresponding methods. + #[allow(clippy::derive_partial_eq_without_eq)] + #[derive(Clone, PartialEq, ::prost::Oneof)] + pub enum MessageRequest { + /// Find a proto file by the file name. + #[prost(string, tag = "3")] + FileByFilename(::prost::alloc::string::String), + /// Find the proto file that declares the given fully-qualified symbol name. + /// This field should be a fully-qualified symbol name + /// (e.g. .\[.\] or .). + #[prost(string, tag = "4")] + FileContainingSymbol(::prost::alloc::string::String), + /// Find the proto file which defines an extension extending the given + /// message type with the given field number. + #[prost(message, tag = "5")] + FileContainingExtension(super::ExtensionRequest), + /// Finds the tag numbers used by all known extensions of the given message + /// type, and appends them to ExtensionNumberResponse in an undefined order. + /// Its corresponding method is best-effort: it's not guaranteed that the + /// reflection service will implement this method, and it's not guaranteed + /// that this method will provide all extensions. Returns + /// StatusCode::UNIMPLEMENTED if it's not implemented. + /// This field should be a fully-qualified type name. The format is + /// . + #[prost(string, tag = "6")] + AllExtensionNumbersOfType(::prost::alloc::string::String), + /// List the full names of registered services. The content will not be + /// checked. + #[prost(string, tag = "7")] + ListServices(::prost::alloc::string::String), + } +} +/// The type name and extension number sent by the client when requesting +/// file_containing_extension. +#[allow(clippy::derive_partial_eq_without_eq)] +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ExtensionRequest { + /// Fully-qualified type name. The format should be . + #[prost(string, tag = "1")] + pub containing_type: ::prost::alloc::string::String, + #[prost(int32, tag = "2")] + pub extension_number: i32, +} +/// The message sent by the server to answer ServerReflectionInfo method. +#[allow(clippy::derive_partial_eq_without_eq)] +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ServerReflectionResponse { + #[prost(string, tag = "1")] + pub valid_host: ::prost::alloc::string::String, + #[prost(message, optional, tag = "2")] + pub original_request: ::core::option::Option, + /// The server sets one of the following fields according to the message_request + /// in the request. + #[prost(oneof = "server_reflection_response::MessageResponse", tags = "4, 5, 6, 7")] + pub message_response: ::core::option::Option< + server_reflection_response::MessageResponse, + >, +} +/// Nested message and enum types in `ServerReflectionResponse`. +pub mod server_reflection_response { + /// The server sets one of the following fields according to the message_request + /// in the request. + #[allow(clippy::derive_partial_eq_without_eq)] + #[derive(Clone, PartialEq, ::prost::Oneof)] + pub enum MessageResponse { + /// This message is used to answer file_by_filename, file_containing_symbol, + /// file_containing_extension requests with transitive dependencies. + /// As the repeated label is not allowed in oneof fields, we use a + /// FileDescriptorResponse message to encapsulate the repeated fields. + /// The reflection service is allowed to avoid sending FileDescriptorProtos + /// that were previously sent in response to earlier requests in the stream. + #[prost(message, tag = "4")] + FileDescriptorResponse(super::FileDescriptorResponse), + /// This message is used to answer all_extension_numbers_of_type requests. + #[prost(message, tag = "5")] + AllExtensionNumbersResponse(super::ExtensionNumberResponse), + /// This message is used to answer list_services requests. + #[prost(message, tag = "6")] + ListServicesResponse(super::ListServiceResponse), + /// This message is used when an error occurs. + #[prost(message, tag = "7")] + ErrorResponse(super::ErrorResponse), + } +} +/// Serialized FileDescriptorProto messages sent by the server answering +/// a file_by_filename, file_containing_symbol, or file_containing_extension +/// request. +#[allow(clippy::derive_partial_eq_without_eq)] +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct FileDescriptorResponse { + /// Serialized FileDescriptorProto messages. We avoid taking a dependency on + /// descriptor.proto, which uses proto2 only features, by making them opaque + /// bytes instead. + #[prost(bytes = "vec", repeated, tag = "1")] + pub file_descriptor_proto: ::prost::alloc::vec::Vec<::prost::alloc::vec::Vec>, +} +/// A list of extension numbers sent by the server answering +/// all_extension_numbers_of_type request. +#[allow(clippy::derive_partial_eq_without_eq)] +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ExtensionNumberResponse { + /// Full name of the base type, including the package name. The format + /// is . + #[prost(string, tag = "1")] + pub base_type_name: ::prost::alloc::string::String, + #[prost(int32, repeated, tag = "2")] + pub extension_number: ::prost::alloc::vec::Vec, +} +/// A list of ServiceResponse sent by the server answering list_services request. +#[allow(clippy::derive_partial_eq_without_eq)] +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ListServiceResponse { + /// The information of each service may be expanded in the future, so we use + /// ServiceResponse message to encapsulate it. + #[prost(message, repeated, tag = "1")] + pub service: ::prost::alloc::vec::Vec, +} +/// The information of a single service used by ListServiceResponse to answer +/// list_services request. +#[allow(clippy::derive_partial_eq_without_eq)] +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ServiceResponse { + /// Full name of a registered service, including its package name. The format + /// is . + #[prost(string, tag = "1")] + pub name: ::prost::alloc::string::String, +} +/// The error code and error message sent by the server when an error occurs. +#[allow(clippy::derive_partial_eq_without_eq)] +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct ErrorResponse { + /// This field uses the error codes defined in grpc::StatusCode. + #[prost(int32, tag = "1")] + pub error_code: i32, + #[prost(string, tag = "2")] + pub error_message: ::prost::alloc::string::String, +} +/// Generated client implementations. +pub mod server_reflection_client { + #![allow(unused_variables, dead_code, missing_docs, clippy::let_unit_value)] + use tonic::codegen::*; + use tonic::codegen::http::Uri; + #[derive(Debug, Clone)] + pub struct ServerReflectionClient { + inner: tonic::client::Grpc, + } + impl ServerReflectionClient + where + T: tonic::client::GrpcService, + T::Error: Into, + T::ResponseBody: Body + Send + 'static, + ::Error: Into + Send, + { + pub fn new(inner: T) -> Self { + let inner = tonic::client::Grpc::new(inner); + Self { inner } + } + pub fn with_origin(inner: T, origin: Uri) -> Self { + let inner = tonic::client::Grpc::with_origin(inner, origin); + Self { inner } + } + pub fn with_interceptor( + inner: T, + interceptor: F, + ) -> ServerReflectionClient> + where + F: tonic::service::Interceptor, + T::ResponseBody: Default, + T: tonic::codegen::Service< + http::Request, + Response = http::Response< + >::ResponseBody, + >, + >, + , + >>::Error: Into + Send + Sync, + { + ServerReflectionClient::new(InterceptedService::new(inner, interceptor)) + } + /// Compress requests with the given encoding. + /// + /// This requires the server to support it otherwise it might respond with an + /// error. + #[must_use] + pub fn send_compressed(mut self, encoding: CompressionEncoding) -> Self { + self.inner = self.inner.send_compressed(encoding); + self + } + /// Enable decompressing responses. + #[must_use] + pub fn accept_compressed(mut self, encoding: CompressionEncoding) -> Self { + self.inner = self.inner.accept_compressed(encoding); + self + } + /// Limits the maximum size of a decoded message. + /// + /// Default: `4MB` + #[must_use] + pub fn max_decoding_message_size(mut self, limit: usize) -> Self { + self.inner = self.inner.max_decoding_message_size(limit); + self + } + /// Limits the maximum size of an encoded message. + /// + /// Default: `usize::MAX` + #[must_use] + pub fn max_encoding_message_size(mut self, limit: usize) -> Self { + self.inner = self.inner.max_encoding_message_size(limit); + self + } + /// The reflection service is structured as a bidirectional stream, ensuring + /// all related requests go to a single server. + pub async fn server_reflection_info( + &mut self, + request: impl tonic::IntoStreamingRequest< + Message = super::ServerReflectionRequest, + >, + ) -> std::result::Result< + tonic::Response>, + tonic::Status, + > { + self.inner + .ready() + .await + .map_err(|e| { + tonic::Status::new( + tonic::Code::Unknown, + format!("Service was not ready: {}", e.into()), + ) + })?; + let codec = tonic::codec::ProstCodec::default(); + let path = http::uri::PathAndQuery::from_static( + "/grpc.reflection.v1.ServerReflection/ServerReflectionInfo", + ); + let mut req = request.into_streaming_request(); + req.extensions_mut() + .insert( + GrpcMethod::new( + "grpc.reflection.v1.ServerReflection", + "ServerReflectionInfo", + ), + ); + self.inner.streaming(req, path, codec).await + } + } +} +/// Generated server implementations. +pub mod server_reflection_server { + #![allow(unused_variables, dead_code, missing_docs, clippy::let_unit_value)] + use tonic::codegen::*; + /// Generated trait containing gRPC methods that should be implemented for use with ServerReflectionServer. + #[async_trait] + pub trait ServerReflection: Send + Sync + 'static { + /// Server streaming response type for the ServerReflectionInfo method. + type ServerReflectionInfoStream: tonic::codegen::tokio_stream::Stream< + Item = std::result::Result< + super::ServerReflectionResponse, + tonic::Status, + >, + > + + Send + + 'static; + /// The reflection service is structured as a bidirectional stream, ensuring + /// all related requests go to a single server. + async fn server_reflection_info( + &self, + request: tonic::Request>, + ) -> std::result::Result< + tonic::Response, + tonic::Status, + >; + } + #[derive(Debug)] + pub struct ServerReflectionServer { + inner: _Inner, + accept_compression_encodings: EnabledCompressionEncodings, + send_compression_encodings: EnabledCompressionEncodings, + max_decoding_message_size: Option, + max_encoding_message_size: Option, + } + struct _Inner(Arc); + impl ServerReflectionServer { + pub fn new(inner: T) -> Self { + Self::from_arc(Arc::new(inner)) + } + pub fn from_arc(inner: Arc) -> Self { + let inner = _Inner(inner); + Self { + inner, + accept_compression_encodings: Default::default(), + send_compression_encodings: Default::default(), + max_decoding_message_size: None, + max_encoding_message_size: None, + } + } + pub fn with_interceptor( + inner: T, + interceptor: F, + ) -> InterceptedService + where + F: tonic::service::Interceptor, + { + InterceptedService::new(Self::new(inner), interceptor) + } + /// Enable decompressing requests with the given encoding. + #[must_use] + pub fn accept_compressed(mut self, encoding: CompressionEncoding) -> Self { + self.accept_compression_encodings.enable(encoding); + self + } + /// Compress responses with the given encoding, if the client supports it. + #[must_use] + pub fn send_compressed(mut self, encoding: CompressionEncoding) -> Self { + self.send_compression_encodings.enable(encoding); + self + } + /// Limits the maximum size of a decoded message. + /// + /// Default: `4MB` + #[must_use] + pub fn max_decoding_message_size(mut self, limit: usize) -> Self { + self.max_decoding_message_size = Some(limit); + self + } + /// Limits the maximum size of an encoded message. + /// + /// Default: `usize::MAX` + #[must_use] + pub fn max_encoding_message_size(mut self, limit: usize) -> Self { + self.max_encoding_message_size = Some(limit); + self + } + } + impl tonic::codegen::Service> for ServerReflectionServer + where + T: ServerReflection, + B: Body + Send + 'static, + B::Error: Into + Send + 'static, + { + type Response = http::Response; + type Error = std::convert::Infallible; + type Future = BoxFuture; + fn poll_ready( + &mut self, + _cx: &mut Context<'_>, + ) -> Poll> { + Poll::Ready(Ok(())) + } + fn call(&mut self, req: http::Request) -> Self::Future { + let inner = self.inner.clone(); + match req.uri().path() { + "/grpc.reflection.v1.ServerReflection/ServerReflectionInfo" => { + #[allow(non_camel_case_types)] + struct ServerReflectionInfoSvc(pub Arc); + impl< + T: ServerReflection, + > tonic::server::StreamingService + for ServerReflectionInfoSvc { + type Response = super::ServerReflectionResponse; + type ResponseStream = T::ServerReflectionInfoStream; + type Future = BoxFuture< + tonic::Response, + tonic::Status, + >; + fn call( + &mut self, + request: tonic::Request< + tonic::Streaming, + >, + ) -> Self::Future { + let inner = Arc::clone(&self.0); + let fut = async move { + ::server_reflection_info( + &inner, + request, + ) + .await + }; + Box::pin(fut) + } + } + let accept_compression_encodings = self.accept_compression_encodings; + let send_compression_encodings = self.send_compression_encodings; + let max_decoding_message_size = self.max_decoding_message_size; + let max_encoding_message_size = self.max_encoding_message_size; + let inner = self.inner.clone(); + let fut = async move { + let inner = inner.0; + let method = ServerReflectionInfoSvc(inner); + let codec = tonic::codec::ProstCodec::default(); + let mut grpc = tonic::server::Grpc::new(codec) + .apply_compression_config( + accept_compression_encodings, + send_compression_encodings, + ) + .apply_max_message_size_config( + max_decoding_message_size, + max_encoding_message_size, + ); + let res = grpc.streaming(method, req).await; + Ok(res) + }; + Box::pin(fut) + } + _ => { + Box::pin(async move { + Ok( + http::Response::builder() + .status(200) + .header("grpc-status", "12") + .header("content-type", "application/grpc") + .body(empty_body()) + .unwrap(), + ) + }) + } + } + } + } + impl Clone for ServerReflectionServer { + fn clone(&self) -> Self { + let inner = self.inner.clone(); + Self { + inner, + accept_compression_encodings: self.accept_compression_encodings, + send_compression_encodings: self.send_compression_encodings, + max_decoding_message_size: self.max_decoding_message_size, + max_encoding_message_size: self.max_encoding_message_size, + } + } + } + impl Clone for _Inner { + fn clone(&self) -> Self { + Self(Arc::clone(&self.0)) + } + } + impl std::fmt::Debug for _Inner { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + write!(f, "{:?}", self.0) + } + } + impl tonic::server::NamedService for ServerReflectionServer { + const NAME: &'static str = "grpc.reflection.v1.ServerReflection"; + } +} diff --git a/tonic-reflection/src/generated/reflection_v1.bin b/tonic-reflection/src/generated/reflection_v1.bin new file mode 100644 index 0000000000000000000000000000000000000000..4b4ecaef9d7cc999759a06523a81fdb2b6fdbca3 GIT binary patch literal 7639 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 Builder<'b> { /// Build a gRPC Reflection Service to be served via Tonic. pub fn build(mut self) -> Result, Error> { if self.include_reflection_service { - self = self.register_encoded_file_descriptor_set(crate::pb::FILE_DESCRIPTOR_SET); + self = self.register_encoded_file_descriptor_set(crate::pb::v1::FILE_DESCRIPTOR_SET); } for encoded in &self.encoded_file_descriptor_sets { diff --git a/tonic-reflection/tests/server.rs b/tonic-reflection/tests/server.rs index ab210d5db..facb75034 100644 --- a/tonic-reflection/tests/server.rs +++ b/tonic-reflection/tests/server.rs @@ -4,7 +4,7 @@ use tokio::sync::oneshot; use tokio_stream::{wrappers::TcpListenerStream, StreamExt}; use tonic::{transport::Server, Request}; use tonic_reflection::{ - pb::{ + pb::v1::{ server_reflection_client::ServerReflectionClient, server_reflection_request::MessageRequest, server_reflection_response::MessageResponse, ServerReflectionRequest, ServiceResponse, FILE_DESCRIPTOR_SET, @@ -34,7 +34,7 @@ async fn test_list_services() { assert_eq!( services.service, vec![ServiceResponse { - name: String::from("grpc.reflection.v1alpha.ServerReflection") + name: String::from("grpc.reflection.v1.ServerReflection") }] ); } else { @@ -47,7 +47,7 @@ async fn test_file_by_filename() { let response = make_test_reflection_request(ServerReflectionRequest { host: "".to_string(), message_request: Some(MessageRequest::FileByFilename(String::from( - "reflection.proto", + "reflection_v1.proto", ))), }) .await; @@ -71,7 +71,7 @@ async fn test_file_containing_symbol() { let response = make_test_reflection_request(ServerReflectionRequest { host: "".to_string(), message_request: Some(MessageRequest::FileContainingSymbol(String::from( - "grpc.reflection.v1alpha.ServerReflection", + "grpc.reflection.v1.ServerReflection", ))), }) .await; diff --git a/tonic-types/src/generated/types.bin b/tonic-types/src/generated/types.bin index 3b6d259bd2462fa78dc372f61dd73c045b19715b..8af4e92be3100c6c490d5ddaa2f1935a01022829 100644 GIT binary patch delta 279 zcmeC#%lL6GBMa9NJ+4XOQWND4ao*M8Vi#hSV_3NHg^SOza|-U^=3-%CU}6wr znaju|CbgD(^L>FUEcFVG#R~ap3ciVz3PuJ-#yScm8L3673W-3Rm#?7T)bSYAm=fH+{Y;h<1&M|BBEU3V8;U` h7fg;9ii