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