// // // 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. // // #include "src/cpp/ext/proto_server_reflection.h" #include #include #include #include #include // IWYU pragma: no_include "google/protobuf/descriptor.h" // IWYU pragma: no_include // IWYU pragma: no_include "src/proto/grpc/reflection/v1/reflection.pb.h" // IWYU pragma: no_include "src/proto/grpc/reflection/v1alpha/reflection.pb.h" namespace grpc { template Status ProtoServerReflectionBackend::ServerReflectionInfo( ServerReaderWriter* stream) const { Request request; Response response; Status status; while (stream->Read(&request)) { switch (request.message_request_case()) { case Request::MessageRequestCase::kFileByFilename: status = GetFileByName(request.file_by_filename(), &response); break; case Request::MessageRequestCase::kFileContainingSymbol: status = GetFileContainingSymbol(request.file_containing_symbol(), &response); break; case Request::MessageRequestCase::kFileContainingExtension: status = GetFileContainingExtension( &request.file_containing_extension(), &response); break; case Request::MessageRequestCase::kAllExtensionNumbersOfType: status = GetAllExtensionNumbers( request.all_extension_numbers_of_type(), response.mutable_all_extension_numbers_response()); break; case Request::MessageRequestCase::kListServices: status = ListService(response.mutable_list_services_response()); break; default: status = Status(StatusCode::UNIMPLEMENTED, ""); } if (!status.ok()) { FillErrorResponse(status, response.mutable_error_response()); } response.set_valid_host(request.host()); response.set_allocated_original_request(new Request(request)); stream->Write(response); } return Status::OK; } template void ProtoServerReflectionBackend::FillErrorResponse( const Status& status, Response* error_response) const { error_response->set_error_code(status.error_code()); error_response->set_error_message(status.error_message()); } template Status ProtoServerReflectionBackend::ListService(Response* response) const { if (services_ == nullptr) { return Status(StatusCode::NOT_FOUND, "Services not found."); } for (const auto& value : *services_) { auto* service_response = response->add_service(); service_response->set_name(value); } return Status::OK; } template Status ProtoServerReflectionBackend::GetFileByName(const std::string& file_name, Response* response) const { if (descriptor_pool_ == nullptr) { return Status::CANCELLED; } const protobuf::FileDescriptor* file_desc = descriptor_pool_->FindFileByName(file_name); if (file_desc == nullptr) { return Status(StatusCode::NOT_FOUND, "File not found."); } std::unordered_set seen_files; FillFileDescriptorResponse(file_desc, response, &seen_files); return Status::OK; } template Status ProtoServerReflectionBackend::GetFileContainingSymbol( const std::string& symbol, Response* response) const { if (descriptor_pool_ == nullptr) { return Status::CANCELLED; } const protobuf::FileDescriptor* file_desc = descriptor_pool_->FindFileContainingSymbol(symbol); if (file_desc == nullptr) { return Status(StatusCode::NOT_FOUND, "Symbol not found."); } std::unordered_set seen_files; FillFileDescriptorResponse(file_desc, response, &seen_files); return Status::OK; } template Status ProtoServerReflectionBackend::GetFileContainingExtension( const Request* request, Response* response) const { if (descriptor_pool_ == nullptr) { return Status::CANCELLED; } const protobuf::Descriptor* desc = descriptor_pool_->FindMessageTypeByName(request->containing_type()); if (desc == nullptr) { return Status(StatusCode::NOT_FOUND, "Type not found."); } const protobuf::FieldDescriptor* field_desc = descriptor_pool_->FindExtensionByNumber(desc, request->extension_number()); if (field_desc == nullptr) { return Status(StatusCode::NOT_FOUND, "Extension not found."); } std::unordered_set seen_files; FillFileDescriptorResponse(field_desc->file(), response, &seen_files); return Status::OK; } template Status ProtoServerReflectionBackend::GetAllExtensionNumbers( const std::string& type, Response* response) const { if (descriptor_pool_ == nullptr) { return Status::CANCELLED; } const protobuf::Descriptor* desc = descriptor_pool_->FindMessageTypeByName(type); if (desc == nullptr) { return Status(StatusCode::NOT_FOUND, "Type not found."); } std::vector extensions; descriptor_pool_->FindAllExtensions(desc, &extensions); for (const auto& value : extensions) { response->add_extension_number(value->number()); } response->set_base_type_name(type); return Status::OK; } template void ProtoServerReflectionBackend::FillFileDescriptorResponse( const protobuf::FileDescriptor* file_desc, Response* response, std::unordered_set* seen_files) const { if (seen_files->find(file_desc->name()) != seen_files->end()) { return; } seen_files->insert(file_desc->name()); protobuf::FileDescriptorProto file_desc_proto; std::string data; file_desc->CopyTo(&file_desc_proto); file_desc_proto.SerializeToString(&data); response->mutable_file_descriptor_response()->add_file_descriptor_proto(data); for (int i = 0; i < file_desc->dependency_count(); ++i) { FillFileDescriptorResponse(file_desc->dependency(i), response, seen_files); } } Status ProtoServerReflection::ServerReflectionInfo( ServerContext* /* context */, ServerReaderWriter* stream) { return backend_->ServerReflectionInfo(stream); } Status ProtoServerReflectionV1::ServerReflectionInfo( ServerContext* /* context */, ServerReaderWriter* stream) { return backend_->ServerReflectionInfo(stream); } } // namespace grpc