mirror of https://github.com/grpc/grpc.git
The C based gRPC (C++, Python, Ruby, Objective-C, PHP, C#)
https://grpc.io/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
204 lines
7.2 KiB
204 lines
7.2 KiB
// |
|
// Copyright 2022 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 <memory> |
|
|
|
#include <gmock/gmock.h> |
|
#include <gtest/gtest.h> |
|
|
|
#include "absl/strings/str_cat.h" |
|
#include "absl/strings/str_format.h" |
|
|
|
#include <grpcpp/create_channel.h> |
|
#include <grpcpp/security/credentials.h> |
|
#include <grpcpp/server_builder.h> |
|
#include <grpcpp/support/status.h> |
|
|
|
#include "src/core/lib/gprpp/crash.h" |
|
#include "src/core/lib/gprpp/host_port.h" |
|
#include "test/core/util/port.h" |
|
#include "test/core/util/test_config.h" |
|
#include "test/cpp/interop/istio_echo_server_lib.h" |
|
|
|
namespace grpc { |
|
namespace testing { |
|
namespace { |
|
|
|
using proto::EchoRequest; |
|
using proto::EchoResponse; |
|
using proto::EchoTestService; |
|
using proto::ForwardEchoRequest; |
|
using proto::ForwardEchoResponse; |
|
|
|
// A very simple EchoTestService implementation that just echoes back the |
|
// message without handling any other expectations for ForwardEcho. |
|
class SimpleEchoTestServerImpl : public proto::EchoTestService::Service { |
|
public: |
|
explicit SimpleEchoTestServerImpl() {} |
|
|
|
grpc::Status Echo(grpc::ServerContext* /* context */, |
|
const proto::EchoRequest* /* request */, |
|
proto::EchoResponse* /* response */) override { |
|
grpc_core::Crash("unreachable"); |
|
return Status(StatusCode::INVALID_ARGUMENT, "Unexpected"); |
|
} |
|
|
|
grpc::Status ForwardEcho(grpc::ServerContext* /*context*/, |
|
const proto::ForwardEchoRequest* request, |
|
proto::ForwardEchoResponse* response) override { |
|
if (fail_rpc_) { |
|
return Status(StatusCode::UNAVAILABLE, "fail rpc"); |
|
} |
|
response->add_output(request->message()); |
|
return Status::OK; |
|
} |
|
|
|
void set_fail_rpc(bool fail_rpc) { fail_rpc_ = fail_rpc; } |
|
|
|
private: |
|
std::string hostname_; |
|
std::string forwarding_address_; |
|
std::atomic<bool> fail_rpc_{false}; |
|
// The following fields are not set yet. But we may need them later. |
|
// int port_; |
|
// std::string version_; |
|
// std::string cluster_; |
|
// std::string istio_version_; |
|
}; |
|
|
|
class EchoTest : public ::testing::Test { |
|
protected: |
|
EchoTest() { |
|
// Start the simple server which will handle protocols that |
|
// EchoTestServiceImpl does not handle. |
|
int forwarding_port = grpc_pick_unused_port_or_die(); |
|
forwarding_address_ = grpc_core::JoinHostPort("localhost", forwarding_port); |
|
ServerBuilder simple_builder; |
|
simple_builder.RegisterService(&simple_test_service_impl_); |
|
simple_builder.AddListeningPort(forwarding_address_, |
|
InsecureServerCredentials()); |
|
simple_server_ = simple_builder.BuildAndStart(); |
|
// Start the EchoTestServiceImpl server |
|
ServerBuilder builder; |
|
echo_test_service_impl_ = std::make_unique<EchoTestServiceImpl>( |
|
"hostname", "v1", forwarding_address_); |
|
builder.RegisterService(echo_test_service_impl_.get()); |
|
int port = grpc_pick_unused_port_or_die(); |
|
server_address_ = grpc_core::JoinHostPort("localhost", port); |
|
builder.AddListeningPort(server_address_, InsecureServerCredentials()); |
|
server_ = builder.BuildAndStart(); |
|
|
|
auto channel = CreateChannel(server_address_, InsecureChannelCredentials()); |
|
stub_ = EchoTestService::NewStub(channel); |
|
} |
|
|
|
std::string forwarding_address_; |
|
SimpleEchoTestServerImpl simple_test_service_impl_; |
|
std::unique_ptr<EchoTestServiceImpl> echo_test_service_impl_; |
|
std::string server_address_; |
|
std::unique_ptr<Server> server_; |
|
std::unique_ptr<Server> simple_server_; |
|
std::unique_ptr<EchoTestService::Stub> stub_; |
|
}; |
|
|
|
TEST_F(EchoTest, SimpleEchoTest) { |
|
ClientContext context; |
|
EchoRequest request; |
|
EchoResponse response; |
|
request.set_message("hello"); |
|
auto status = stub_->Echo(&context, request, &response); |
|
ASSERT_TRUE(status.ok()); |
|
EXPECT_THAT(response.message(), |
|
::testing::AllOf(::testing::HasSubstr("StatusCode=200\n"), |
|
::testing::HasSubstr("Hostname=hostname\n"), |
|
::testing::HasSubstr("Echo=hello\n"), |
|
::testing::HasSubstr("Host="), |
|
::testing::HasSubstr("IP="), |
|
::testing::HasSubstr("ServiceVersion=v1"))); |
|
} |
|
|
|
TEST_F(EchoTest, ForwardEchoTest) { |
|
ClientContext context; |
|
ForwardEchoRequest request; |
|
ForwardEchoResponse response; |
|
request.set_count(3); |
|
request.set_qps(1); |
|
request.set_timeout_micros(20 * 1000 * 1000); // 20 seconds |
|
request.set_url(absl::StrCat("grpc://", server_address_)); |
|
request.set_message("hello"); |
|
auto status = stub_->ForwardEcho(&context, request, &response); |
|
ASSERT_TRUE(status.ok()); |
|
for (int i = 0; i < 3; ++i) { |
|
EXPECT_THAT( |
|
response.output()[i], |
|
::testing::AllOf( |
|
::testing::HasSubstr( |
|
absl::StrFormat("[%d body] StatusCode=200\n", i)), |
|
::testing::HasSubstr( |
|
absl::StrFormat("[%d body] Hostname=hostname\n", i)), |
|
::testing::HasSubstr(absl::StrFormat("[%d body] Echo=hello\n", i)), |
|
::testing::HasSubstr(absl::StrFormat("[%d body] Host=", i)), |
|
::testing::HasSubstr( |
|
absl::StrFormat("[%d body] ServiceVersion=v1", i)))); |
|
} |
|
} |
|
|
|
TEST_F(EchoTest, ForwardEchoTestUnhandledProtocols) { |
|
ClientContext context; |
|
ForwardEchoRequest request; |
|
ForwardEchoResponse response; |
|
request.set_count(3); |
|
request.set_qps(1); |
|
request.set_timeout_micros(20 * 1000 * 1000); // 20 seconds |
|
// http protocol is unhandled by EchoTestServiceImpl and should be forwarded |
|
// to SimpleEchoTestServiceImpl |
|
request.set_url(absl::StrCat("http://", server_address_)); |
|
request.set_message("hello"); |
|
auto status = stub_->ForwardEcho(&context, request, &response); |
|
ASSERT_TRUE(status.ok()) << "Code = " << status.error_code() |
|
<< " Message = " << status.error_message(); |
|
ASSERT_FALSE(response.output().empty()); |
|
EXPECT_EQ(response.output()[0], "hello"); |
|
} |
|
|
|
TEST_F(EchoTest, ForwardEchoFailure) { |
|
simple_test_service_impl_.set_fail_rpc(true); |
|
ClientContext context; |
|
ForwardEchoRequest request; |
|
ForwardEchoResponse response; |
|
request.set_count(3); |
|
request.set_qps(1); |
|
request.set_timeout_micros(20 * 1000 * 1000); // 20 seconds |
|
// Use the unhandled protocol to make sure that we forward the request to |
|
// SimpleEchoTestServerImpl. |
|
request.set_url(absl::StrCat("http://", server_address_)); |
|
request.set_message("hello"); |
|
auto status = stub_->ForwardEcho(&context, request, &response); |
|
ASSERT_EQ(status.error_code(), StatusCode::UNAVAILABLE); |
|
} |
|
|
|
} // namespace |
|
} // namespace testing |
|
} // namespace grpc |
|
|
|
int main(int argc, char** argv) { |
|
::testing::InitGoogleTest(&argc, argv); |
|
grpc::testing::TestEnvironment env(&argc, argv); |
|
grpc_init(); |
|
auto result = RUN_ALL_TESTS(); |
|
grpc_shutdown(); |
|
return result; |
|
}
|
|
|