xref: /aosp_15_r20/external/grpc-grpc/test/cpp/interop/http2_client.cc (revision cc02d7e222339f7a4f6ba5f422e6413f4bd931f2)
1*cc02d7e2SAndroid Build Coastguard Worker //
2*cc02d7e2SAndroid Build Coastguard Worker //
3*cc02d7e2SAndroid Build Coastguard Worker // Copyright 2016 gRPC authors.
4*cc02d7e2SAndroid Build Coastguard Worker //
5*cc02d7e2SAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
6*cc02d7e2SAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
7*cc02d7e2SAndroid Build Coastguard Worker // You may obtain a copy of the License at
8*cc02d7e2SAndroid Build Coastguard Worker //
9*cc02d7e2SAndroid Build Coastguard Worker //     http://www.apache.org/licenses/LICENSE-2.0
10*cc02d7e2SAndroid Build Coastguard Worker //
11*cc02d7e2SAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
12*cc02d7e2SAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
13*cc02d7e2SAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14*cc02d7e2SAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
15*cc02d7e2SAndroid Build Coastguard Worker // limitations under the License.
16*cc02d7e2SAndroid Build Coastguard Worker //
17*cc02d7e2SAndroid Build Coastguard Worker //
18*cc02d7e2SAndroid Build Coastguard Worker 
19*cc02d7e2SAndroid Build Coastguard Worker #include "test/cpp/interop/http2_client.h"
20*cc02d7e2SAndroid Build Coastguard Worker 
21*cc02d7e2SAndroid Build Coastguard Worker #include <thread>
22*cc02d7e2SAndroid Build Coastguard Worker 
23*cc02d7e2SAndroid Build Coastguard Worker #include "absl/flags/flag.h"
24*cc02d7e2SAndroid Build Coastguard Worker #include "absl/strings/str_format.h"
25*cc02d7e2SAndroid Build Coastguard Worker 
26*cc02d7e2SAndroid Build Coastguard Worker #include <grpc/support/alloc.h>
27*cc02d7e2SAndroid Build Coastguard Worker #include <grpc/support/log.h>
28*cc02d7e2SAndroid Build Coastguard Worker #include <grpcpp/channel.h>
29*cc02d7e2SAndroid Build Coastguard Worker #include <grpcpp/client_context.h>
30*cc02d7e2SAndroid Build Coastguard Worker 
31*cc02d7e2SAndroid Build Coastguard Worker #include "src/core/lib/gpr/string.h"
32*cc02d7e2SAndroid Build Coastguard Worker #include "src/core/lib/gpr/useful.h"
33*cc02d7e2SAndroid Build Coastguard Worker #include "src/core/lib/gprpp/crash.h"
34*cc02d7e2SAndroid Build Coastguard Worker #include "src/proto/grpc/testing/messages.pb.h"
35*cc02d7e2SAndroid Build Coastguard Worker #include "src/proto/grpc/testing/test.grpc.pb.h"
36*cc02d7e2SAndroid Build Coastguard Worker #include "test/cpp/util/create_test_channel.h"
37*cc02d7e2SAndroid Build Coastguard Worker #include "test/cpp/util/test_config.h"
38*cc02d7e2SAndroid Build Coastguard Worker 
39*cc02d7e2SAndroid Build Coastguard Worker namespace grpc {
40*cc02d7e2SAndroid Build Coastguard Worker namespace testing {
41*cc02d7e2SAndroid Build Coastguard Worker 
42*cc02d7e2SAndroid Build Coastguard Worker namespace {
43*cc02d7e2SAndroid Build Coastguard Worker const int kLargeRequestSize = 271828;
44*cc02d7e2SAndroid Build Coastguard Worker const int kLargeResponseSize = 314159;
45*cc02d7e2SAndroid Build Coastguard Worker }  // namespace
46*cc02d7e2SAndroid Build Coastguard Worker 
ServiceStub(const std::shared_ptr<Channel> & channel)47*cc02d7e2SAndroid Build Coastguard Worker Http2Client::ServiceStub::ServiceStub(const std::shared_ptr<Channel>& channel)
48*cc02d7e2SAndroid Build Coastguard Worker     : channel_(channel) {
49*cc02d7e2SAndroid Build Coastguard Worker   stub_ = TestService::NewStub(channel);
50*cc02d7e2SAndroid Build Coastguard Worker }
51*cc02d7e2SAndroid Build Coastguard Worker 
Get()52*cc02d7e2SAndroid Build Coastguard Worker TestService::Stub* Http2Client::ServiceStub::Get() { return stub_.get(); }
53*cc02d7e2SAndroid Build Coastguard Worker 
Http2Client(const std::shared_ptr<Channel> & channel)54*cc02d7e2SAndroid Build Coastguard Worker Http2Client::Http2Client(const std::shared_ptr<Channel>& channel)
55*cc02d7e2SAndroid Build Coastguard Worker     : serviceStub_(channel),
56*cc02d7e2SAndroid Build Coastguard Worker       channel_(channel),
57*cc02d7e2SAndroid Build Coastguard Worker       defaultRequest_(BuildDefaultRequest()) {}
58*cc02d7e2SAndroid Build Coastguard Worker 
AssertStatusCode(const Status & s,StatusCode expected_code)59*cc02d7e2SAndroid Build Coastguard Worker bool Http2Client::AssertStatusCode(const Status& s, StatusCode expected_code) {
60*cc02d7e2SAndroid Build Coastguard Worker   if (s.error_code() == expected_code) {
61*cc02d7e2SAndroid Build Coastguard Worker     return true;
62*cc02d7e2SAndroid Build Coastguard Worker   }
63*cc02d7e2SAndroid Build Coastguard Worker 
64*cc02d7e2SAndroid Build Coastguard Worker   grpc_core::Crash(absl::StrFormat(
65*cc02d7e2SAndroid Build Coastguard Worker       "Error status code: %d (expected: %d), message: %s", s.error_code(),
66*cc02d7e2SAndroid Build Coastguard Worker       expected_code, s.error_message().c_str()));
67*cc02d7e2SAndroid Build Coastguard Worker }
68*cc02d7e2SAndroid Build Coastguard Worker 
SendUnaryCall(SimpleResponse * response)69*cc02d7e2SAndroid Build Coastguard Worker Status Http2Client::SendUnaryCall(SimpleResponse* response) {
70*cc02d7e2SAndroid Build Coastguard Worker   ClientContext context;
71*cc02d7e2SAndroid Build Coastguard Worker   return serviceStub_.Get()->UnaryCall(&context, defaultRequest_, response);
72*cc02d7e2SAndroid Build Coastguard Worker }
73*cc02d7e2SAndroid Build Coastguard Worker 
BuildDefaultRequest()74*cc02d7e2SAndroid Build Coastguard Worker SimpleRequest Http2Client::BuildDefaultRequest() {
75*cc02d7e2SAndroid Build Coastguard Worker   SimpleRequest request;
76*cc02d7e2SAndroid Build Coastguard Worker   request.set_response_size(kLargeResponseSize);
77*cc02d7e2SAndroid Build Coastguard Worker   std::string payload(kLargeRequestSize, '\0');
78*cc02d7e2SAndroid Build Coastguard Worker   request.mutable_payload()->set_body(payload.c_str(), kLargeRequestSize);
79*cc02d7e2SAndroid Build Coastguard Worker   return request;
80*cc02d7e2SAndroid Build Coastguard Worker }
81*cc02d7e2SAndroid Build Coastguard Worker 
DoRstAfterHeader()82*cc02d7e2SAndroid Build Coastguard Worker bool Http2Client::DoRstAfterHeader() {
83*cc02d7e2SAndroid Build Coastguard Worker   gpr_log(GPR_DEBUG, "Sending RPC and expecting reset stream after header");
84*cc02d7e2SAndroid Build Coastguard Worker 
85*cc02d7e2SAndroid Build Coastguard Worker   SimpleResponse response;
86*cc02d7e2SAndroid Build Coastguard Worker   AssertStatusCode(SendUnaryCall(&response), grpc::StatusCode::INTERNAL);
87*cc02d7e2SAndroid Build Coastguard Worker   GPR_ASSERT(!response.has_payload());  // no data should be received
88*cc02d7e2SAndroid Build Coastguard Worker 
89*cc02d7e2SAndroid Build Coastguard Worker   gpr_log(GPR_DEBUG, "Done testing reset stream after header");
90*cc02d7e2SAndroid Build Coastguard Worker   return true;
91*cc02d7e2SAndroid Build Coastguard Worker }
92*cc02d7e2SAndroid Build Coastguard Worker 
DoRstAfterData()93*cc02d7e2SAndroid Build Coastguard Worker bool Http2Client::DoRstAfterData() {
94*cc02d7e2SAndroid Build Coastguard Worker   gpr_log(GPR_DEBUG, "Sending RPC and expecting reset stream after data");
95*cc02d7e2SAndroid Build Coastguard Worker 
96*cc02d7e2SAndroid Build Coastguard Worker   SimpleResponse response;
97*cc02d7e2SAndroid Build Coastguard Worker   AssertStatusCode(SendUnaryCall(&response), grpc::StatusCode::INTERNAL);
98*cc02d7e2SAndroid Build Coastguard Worker   // There is no guarantee that data would be received.
99*cc02d7e2SAndroid Build Coastguard Worker 
100*cc02d7e2SAndroid Build Coastguard Worker   gpr_log(GPR_DEBUG, "Done testing reset stream after data");
101*cc02d7e2SAndroid Build Coastguard Worker   return true;
102*cc02d7e2SAndroid Build Coastguard Worker }
103*cc02d7e2SAndroid Build Coastguard Worker 
DoRstDuringData()104*cc02d7e2SAndroid Build Coastguard Worker bool Http2Client::DoRstDuringData() {
105*cc02d7e2SAndroid Build Coastguard Worker   gpr_log(GPR_DEBUG, "Sending RPC and expecting reset stream during data");
106*cc02d7e2SAndroid Build Coastguard Worker 
107*cc02d7e2SAndroid Build Coastguard Worker   SimpleResponse response;
108*cc02d7e2SAndroid Build Coastguard Worker   AssertStatusCode(SendUnaryCall(&response), grpc::StatusCode::INTERNAL);
109*cc02d7e2SAndroid Build Coastguard Worker   GPR_ASSERT(!response.has_payload());  // no data should be received
110*cc02d7e2SAndroid Build Coastguard Worker 
111*cc02d7e2SAndroid Build Coastguard Worker   gpr_log(GPR_DEBUG, "Done testing reset stream during data");
112*cc02d7e2SAndroid Build Coastguard Worker   return true;
113*cc02d7e2SAndroid Build Coastguard Worker }
114*cc02d7e2SAndroid Build Coastguard Worker 
DoGoaway()115*cc02d7e2SAndroid Build Coastguard Worker bool Http2Client::DoGoaway() {
116*cc02d7e2SAndroid Build Coastguard Worker   gpr_log(GPR_DEBUG, "Sending two RPCs and expecting goaway");
117*cc02d7e2SAndroid Build Coastguard Worker   SimpleResponse response;
118*cc02d7e2SAndroid Build Coastguard Worker   AssertStatusCode(SendUnaryCall(&response), grpc::StatusCode::OK);
119*cc02d7e2SAndroid Build Coastguard Worker   GPR_ASSERT(response.payload().body() ==
120*cc02d7e2SAndroid Build Coastguard Worker              std::string(kLargeResponseSize, '\0'));
121*cc02d7e2SAndroid Build Coastguard Worker 
122*cc02d7e2SAndroid Build Coastguard Worker   // Sleep for one second to give time for client to receive goaway frame.
123*cc02d7e2SAndroid Build Coastguard Worker   gpr_timespec sleep_time = gpr_time_add(
124*cc02d7e2SAndroid Build Coastguard Worker       gpr_now(GPR_CLOCK_REALTIME), gpr_time_from_seconds(1, GPR_TIMESPAN));
125*cc02d7e2SAndroid Build Coastguard Worker   gpr_sleep_until(sleep_time);
126*cc02d7e2SAndroid Build Coastguard Worker 
127*cc02d7e2SAndroid Build Coastguard Worker   response.Clear();
128*cc02d7e2SAndroid Build Coastguard Worker   AssertStatusCode(SendUnaryCall(&response), grpc::StatusCode::OK);
129*cc02d7e2SAndroid Build Coastguard Worker   GPR_ASSERT(response.payload().body() ==
130*cc02d7e2SAndroid Build Coastguard Worker              std::string(kLargeResponseSize, '\0'));
131*cc02d7e2SAndroid Build Coastguard Worker   gpr_log(GPR_DEBUG, "Done testing goaway");
132*cc02d7e2SAndroid Build Coastguard Worker   return true;
133*cc02d7e2SAndroid Build Coastguard Worker }
134*cc02d7e2SAndroid Build Coastguard Worker 
DoPing()135*cc02d7e2SAndroid Build Coastguard Worker bool Http2Client::DoPing() {
136*cc02d7e2SAndroid Build Coastguard Worker   gpr_log(GPR_DEBUG, "Sending RPC and expecting ping");
137*cc02d7e2SAndroid Build Coastguard Worker   SimpleResponse response;
138*cc02d7e2SAndroid Build Coastguard Worker   AssertStatusCode(SendUnaryCall(&response), grpc::StatusCode::OK);
139*cc02d7e2SAndroid Build Coastguard Worker   GPR_ASSERT(response.payload().body() ==
140*cc02d7e2SAndroid Build Coastguard Worker              std::string(kLargeResponseSize, '\0'));
141*cc02d7e2SAndroid Build Coastguard Worker   gpr_log(GPR_DEBUG, "Done testing ping");
142*cc02d7e2SAndroid Build Coastguard Worker   return true;
143*cc02d7e2SAndroid Build Coastguard Worker }
144*cc02d7e2SAndroid Build Coastguard Worker 
MaxStreamsWorker(const std::shared_ptr<grpc::Channel> &)145*cc02d7e2SAndroid Build Coastguard Worker void Http2Client::MaxStreamsWorker(
146*cc02d7e2SAndroid Build Coastguard Worker     const std::shared_ptr<grpc::Channel>& /*channel*/) {
147*cc02d7e2SAndroid Build Coastguard Worker   SimpleResponse response;
148*cc02d7e2SAndroid Build Coastguard Worker   AssertStatusCode(SendUnaryCall(&response), grpc::StatusCode::OK);
149*cc02d7e2SAndroid Build Coastguard Worker   GPR_ASSERT(response.payload().body() ==
150*cc02d7e2SAndroid Build Coastguard Worker              std::string(kLargeResponseSize, '\0'));
151*cc02d7e2SAndroid Build Coastguard Worker }
152*cc02d7e2SAndroid Build Coastguard Worker 
DoMaxStreams()153*cc02d7e2SAndroid Build Coastguard Worker bool Http2Client::DoMaxStreams() {
154*cc02d7e2SAndroid Build Coastguard Worker   gpr_log(GPR_DEBUG, "Testing max streams");
155*cc02d7e2SAndroid Build Coastguard Worker 
156*cc02d7e2SAndroid Build Coastguard Worker   // Make an initial call on the channel to ensure the server's max streams
157*cc02d7e2SAndroid Build Coastguard Worker   // setting is received
158*cc02d7e2SAndroid Build Coastguard Worker   SimpleResponse response;
159*cc02d7e2SAndroid Build Coastguard Worker   AssertStatusCode(SendUnaryCall(&response), grpc::StatusCode::OK);
160*cc02d7e2SAndroid Build Coastguard Worker   GPR_ASSERT(response.payload().body() ==
161*cc02d7e2SAndroid Build Coastguard Worker              std::string(kLargeResponseSize, '\0'));
162*cc02d7e2SAndroid Build Coastguard Worker 
163*cc02d7e2SAndroid Build Coastguard Worker   std::vector<std::thread> test_threads;
164*cc02d7e2SAndroid Build Coastguard Worker   test_threads.reserve(10);
165*cc02d7e2SAndroid Build Coastguard Worker   for (int i = 0; i < 10; i++) {
166*cc02d7e2SAndroid Build Coastguard Worker     test_threads.emplace_back(
167*cc02d7e2SAndroid Build Coastguard Worker         std::thread(&Http2Client::MaxStreamsWorker, this, channel_));
168*cc02d7e2SAndroid Build Coastguard Worker   }
169*cc02d7e2SAndroid Build Coastguard Worker 
170*cc02d7e2SAndroid Build Coastguard Worker   for (auto it = test_threads.begin(); it != test_threads.end(); it++) {
171*cc02d7e2SAndroid Build Coastguard Worker     it->join();
172*cc02d7e2SAndroid Build Coastguard Worker   }
173*cc02d7e2SAndroid Build Coastguard Worker 
174*cc02d7e2SAndroid Build Coastguard Worker   gpr_log(GPR_DEBUG, "Done testing max streams");
175*cc02d7e2SAndroid Build Coastguard Worker   return true;
176*cc02d7e2SAndroid Build Coastguard Worker }
177*cc02d7e2SAndroid Build Coastguard Worker 
178*cc02d7e2SAndroid Build Coastguard Worker }  // namespace testing
179*cc02d7e2SAndroid Build Coastguard Worker }  // namespace grpc
180*cc02d7e2SAndroid Build Coastguard Worker 
181*cc02d7e2SAndroid Build Coastguard Worker ABSL_FLAG(int32_t, server_port, 0, "Server port.");
182*cc02d7e2SAndroid Build Coastguard Worker ABSL_FLAG(std::string, server_host, "localhost", "Server host to connect to");
183*cc02d7e2SAndroid Build Coastguard Worker ABSL_FLAG(std::string, test_case, "rst_after_header",
184*cc02d7e2SAndroid Build Coastguard Worker           "Configure different test cases. Valid options are:\n\n"
185*cc02d7e2SAndroid Build Coastguard Worker           "goaway\n"
186*cc02d7e2SAndroid Build Coastguard Worker           "max_streams\n"
187*cc02d7e2SAndroid Build Coastguard Worker           "ping\n"
188*cc02d7e2SAndroid Build Coastguard Worker           "rst_after_data\n"
189*cc02d7e2SAndroid Build Coastguard Worker           "rst_after_header\n"
190*cc02d7e2SAndroid Build Coastguard Worker           "rst_during_data\n");
191*cc02d7e2SAndroid Build Coastguard Worker 
main(int argc,char ** argv)192*cc02d7e2SAndroid Build Coastguard Worker int main(int argc, char** argv) {
193*cc02d7e2SAndroid Build Coastguard Worker   grpc::testing::InitTest(&argc, &argv, true);
194*cc02d7e2SAndroid Build Coastguard Worker   GPR_ASSERT(absl::GetFlag(FLAGS_server_port));
195*cc02d7e2SAndroid Build Coastguard Worker   const int host_port_buf_size = 1024;
196*cc02d7e2SAndroid Build Coastguard Worker   char host_port[host_port_buf_size];
197*cc02d7e2SAndroid Build Coastguard Worker   snprintf(host_port, host_port_buf_size, "%s:%d",
198*cc02d7e2SAndroid Build Coastguard Worker            absl::GetFlag(FLAGS_server_host).c_str(),
199*cc02d7e2SAndroid Build Coastguard Worker            absl::GetFlag(FLAGS_server_port));
200*cc02d7e2SAndroid Build Coastguard Worker   std::shared_ptr<grpc::Channel> channel =
201*cc02d7e2SAndroid Build Coastguard Worker       grpc::CreateTestChannel(host_port, grpc::testing::INSECURE);
202*cc02d7e2SAndroid Build Coastguard Worker   GPR_ASSERT(channel->WaitForConnected(gpr_time_add(
203*cc02d7e2SAndroid Build Coastguard Worker       gpr_now(GPR_CLOCK_REALTIME), gpr_time_from_seconds(300, GPR_TIMESPAN))));
204*cc02d7e2SAndroid Build Coastguard Worker   grpc::testing::Http2Client client(channel);
205*cc02d7e2SAndroid Build Coastguard Worker   gpr_log(GPR_INFO, "Testing case: %s", absl::GetFlag(FLAGS_test_case).c_str());
206*cc02d7e2SAndroid Build Coastguard Worker   int ret = 0;
207*cc02d7e2SAndroid Build Coastguard Worker   if (absl::GetFlag(FLAGS_test_case) == "rst_after_header") {
208*cc02d7e2SAndroid Build Coastguard Worker     client.DoRstAfterHeader();
209*cc02d7e2SAndroid Build Coastguard Worker   } else if (absl::GetFlag(FLAGS_test_case) == "rst_after_data") {
210*cc02d7e2SAndroid Build Coastguard Worker     client.DoRstAfterData();
211*cc02d7e2SAndroid Build Coastguard Worker   } else if (absl::GetFlag(FLAGS_test_case) == "rst_during_data") {
212*cc02d7e2SAndroid Build Coastguard Worker     client.DoRstDuringData();
213*cc02d7e2SAndroid Build Coastguard Worker   } else if (absl::GetFlag(FLAGS_test_case) == "goaway") {
214*cc02d7e2SAndroid Build Coastguard Worker     client.DoGoaway();
215*cc02d7e2SAndroid Build Coastguard Worker   } else if (absl::GetFlag(FLAGS_test_case) == "ping") {
216*cc02d7e2SAndroid Build Coastguard Worker     client.DoPing();
217*cc02d7e2SAndroid Build Coastguard Worker   } else if (absl::GetFlag(FLAGS_test_case) == "max_streams") {
218*cc02d7e2SAndroid Build Coastguard Worker     client.DoMaxStreams();
219*cc02d7e2SAndroid Build Coastguard Worker   } else {
220*cc02d7e2SAndroid Build Coastguard Worker     const char* testcases[] = {
221*cc02d7e2SAndroid Build Coastguard Worker         "goaway",         "max_streams",      "ping",
222*cc02d7e2SAndroid Build Coastguard Worker         "rst_after_data", "rst_after_header", "rst_during_data"};
223*cc02d7e2SAndroid Build Coastguard Worker     char* joined_testcases =
224*cc02d7e2SAndroid Build Coastguard Worker         gpr_strjoin_sep(testcases, GPR_ARRAY_SIZE(testcases), "\n", nullptr);
225*cc02d7e2SAndroid Build Coastguard Worker 
226*cc02d7e2SAndroid Build Coastguard Worker     gpr_log(GPR_ERROR, "Unsupported test case %s. Valid options are\n%s",
227*cc02d7e2SAndroid Build Coastguard Worker             absl::GetFlag(FLAGS_test_case).c_str(), joined_testcases);
228*cc02d7e2SAndroid Build Coastguard Worker     gpr_free(joined_testcases);
229*cc02d7e2SAndroid Build Coastguard Worker     ret = 1;
230*cc02d7e2SAndroid Build Coastguard Worker   }
231*cc02d7e2SAndroid Build Coastguard Worker 
232*cc02d7e2SAndroid Build Coastguard Worker   return ret;
233*cc02d7e2SAndroid Build Coastguard Worker }
234