1 /*
2 *
3 * Copyright 2015 gRPC authors.
4 *
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
8 *
9 * http://www.apache.org/licenses/LICENSE-2.0
10 *
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
16 *
17 */
18
19 #include "test/core/end2end/end2end_tests.h"
20
21 #include <stdio.h>
22 #include <string.h>
23
24 #include <string>
25
26 #include "absl/strings/str_format.h"
27
28 #include <grpc/byte_buffer.h>
29 #include <grpc/byte_buffer_reader.h>
30 #include <grpc/compression.h>
31 #include <grpc/support/alloc.h>
32 #include <grpc/support/log.h>
33 #include <grpc/support/time.h>
34
35 #include "src/core/lib/channel/channel_args.h"
36 #include "src/core/lib/compression/compression_args.h"
37 #include "src/core/lib/surface/call.h"
38 #include "src/core/lib/surface/call_test_only.h"
39 #include "src/core/lib/transport/static_metadata.h"
40 #include "test/core/end2end/cq_verifier.h"
41
tag(intptr_t t)42 static void* tag(intptr_t t) { return (void*)t; }
43
begin_test(grpc_end2end_test_config config,const char * test_name,grpc_channel_args * client_args,grpc_channel_args * server_args,bool decompress_in_core)44 static grpc_end2end_test_fixture begin_test(grpc_end2end_test_config config,
45 const char* test_name,
46 grpc_channel_args* client_args,
47 grpc_channel_args* server_args,
48 bool decompress_in_core) {
49 grpc_end2end_test_fixture f;
50 gpr_log(GPR_INFO, "Running test: %s%s/%s", test_name,
51 decompress_in_core ? "" : "_with_decompression_disabled",
52 config.name);
53 f = config.create_fixture(client_args, server_args);
54 config.init_server(&f, server_args);
55 config.init_client(&f, client_args);
56 return f;
57 }
58
n_seconds_from_now(int n)59 static gpr_timespec n_seconds_from_now(int n) {
60 return grpc_timeout_seconds_to_deadline(n);
61 }
62
five_seconds_from_now(void)63 static gpr_timespec five_seconds_from_now(void) {
64 return n_seconds_from_now(5);
65 }
66
drain_cq(grpc_completion_queue * cq)67 static void drain_cq(grpc_completion_queue* cq) {
68 grpc_event ev;
69 do {
70 ev = grpc_completion_queue_next(cq, five_seconds_from_now(), nullptr);
71 } while (ev.type != GRPC_QUEUE_SHUTDOWN);
72 }
73
shutdown_server(grpc_end2end_test_fixture * f)74 static void shutdown_server(grpc_end2end_test_fixture* f) {
75 if (!f->server) return;
76 grpc_server_shutdown_and_notify(f->server, f->shutdown_cq, tag(1000));
77 GPR_ASSERT(grpc_completion_queue_pluck(f->shutdown_cq, tag(1000),
78 grpc_timeout_seconds_to_deadline(5),
79 nullptr)
80 .type == GRPC_OP_COMPLETE);
81 grpc_server_destroy(f->server);
82 f->server = nullptr;
83 }
84
shutdown_client(grpc_end2end_test_fixture * f)85 static void shutdown_client(grpc_end2end_test_fixture* f) {
86 if (!f->client) return;
87 grpc_channel_destroy(f->client);
88 f->client = nullptr;
89 }
90
end_test(grpc_end2end_test_fixture * f)91 static void end_test(grpc_end2end_test_fixture* f) {
92 shutdown_server(f);
93 shutdown_client(f);
94
95 grpc_completion_queue_shutdown(f->cq);
96 drain_cq(f->cq);
97 grpc_completion_queue_destroy(f->cq);
98 grpc_completion_queue_destroy(f->shutdown_cq);
99 }
100
request_for_disabled_algorithm(grpc_end2end_test_config config,const char * test_name,uint32_t send_flags_bitmask,grpc_compression_algorithm algorithm_to_disable,grpc_compression_algorithm requested_client_compression_algorithm,grpc_status_code expected_error,grpc_metadata * client_metadata,bool decompress_in_core)101 static void request_for_disabled_algorithm(
102 grpc_end2end_test_config config, const char* test_name,
103 uint32_t send_flags_bitmask,
104 grpc_compression_algorithm algorithm_to_disable,
105 grpc_compression_algorithm requested_client_compression_algorithm,
106 grpc_status_code expected_error, grpc_metadata* client_metadata,
107 bool decompress_in_core) {
108 grpc_call* c;
109 grpc_call* s;
110 grpc_slice request_payload_slice;
111 grpc_byte_buffer* request_payload;
112 grpc_channel_args* client_args;
113 grpc_channel_args* server_args;
114 grpc_end2end_test_fixture f;
115 grpc_op ops[6];
116 grpc_op* op;
117 grpc_metadata_array initial_metadata_recv;
118 grpc_metadata_array trailing_metadata_recv;
119 grpc_metadata_array request_metadata_recv;
120 grpc_byte_buffer* request_payload_recv = nullptr;
121 grpc_call_details call_details;
122 grpc_status_code status;
123 grpc_call_error error;
124 grpc_slice details;
125 int was_cancelled = 2;
126 cq_verifier* cqv;
127 char str[1024];
128
129 memset(str, 'x', 1023);
130 str[1023] = '\0';
131 request_payload_slice = grpc_slice_from_copied_string(str);
132 request_payload = grpc_raw_byte_buffer_create(&request_payload_slice, 1);
133
134 client_args = grpc_channel_args_set_channel_default_compression_algorithm(
135 nullptr, requested_client_compression_algorithm);
136 server_args = grpc_channel_args_set_channel_default_compression_algorithm(
137 nullptr, GRPC_COMPRESS_NONE);
138 server_args = grpc_channel_args_compression_algorithm_set_state(
139 &server_args, algorithm_to_disable, false);
140 if (!decompress_in_core) {
141 grpc_arg disable_decompression_in_core_arg =
142 grpc_channel_arg_integer_create(
143 const_cast<char*>(GRPC_ARG_ENABLE_PER_MESSAGE_DECOMPRESSION), 0);
144 grpc_channel_args* old_client_args = client_args;
145 grpc_channel_args* old_server_args = server_args;
146 client_args = grpc_channel_args_copy_and_add(
147 client_args, &disable_decompression_in_core_arg, 1);
148 server_args = grpc_channel_args_copy_and_add(
149 server_args, &disable_decompression_in_core_arg, 1);
150 grpc_channel_args_destroy(old_client_args);
151 grpc_channel_args_destroy(old_server_args);
152 }
153
154 f = begin_test(config, test_name, client_args, server_args,
155 decompress_in_core);
156 cqv = cq_verifier_create(f.cq);
157
158 gpr_timespec deadline = five_seconds_from_now();
159 c = grpc_channel_create_call(f.client, nullptr, GRPC_PROPAGATE_DEFAULTS, f.cq,
160 grpc_slice_from_static_string("/foo"), nullptr,
161 deadline, nullptr);
162 GPR_ASSERT(c);
163
164 grpc_metadata_array_init(&initial_metadata_recv);
165 grpc_metadata_array_init(&trailing_metadata_recv);
166 grpc_metadata_array_init(&request_metadata_recv);
167 grpc_call_details_init(&call_details);
168
169 error =
170 grpc_server_request_call(f.server, &s, &call_details,
171 &request_metadata_recv, f.cq, f.cq, tag(101));
172 GPR_ASSERT(GRPC_CALL_OK == error);
173
174 memset(ops, 0, sizeof(ops));
175 op = ops;
176 op->op = GRPC_OP_SEND_INITIAL_METADATA;
177 if (client_metadata != nullptr) {
178 op->data.send_initial_metadata.count = 1;
179 op->data.send_initial_metadata.metadata = client_metadata;
180 } else {
181 op->data.send_initial_metadata.count = 0;
182 }
183 op->flags = 0;
184 op->reserved = nullptr;
185 op++;
186 op->op = GRPC_OP_SEND_MESSAGE;
187 op->data.send_message.send_message = request_payload;
188 op->flags = send_flags_bitmask;
189 op->reserved = nullptr;
190 op++;
191 op->op = GRPC_OP_SEND_CLOSE_FROM_CLIENT;
192 op->flags = 0;
193 op->reserved = nullptr;
194 op++;
195 op->op = GRPC_OP_RECV_INITIAL_METADATA;
196 op->data.recv_initial_metadata.recv_initial_metadata = &initial_metadata_recv;
197 op->flags = 0;
198 op->reserved = nullptr;
199 op++;
200 op->op = GRPC_OP_RECV_STATUS_ON_CLIENT;
201 op->data.recv_status_on_client.trailing_metadata = &trailing_metadata_recv;
202 op->data.recv_status_on_client.status = &status;
203 op->data.recv_status_on_client.status_details = &details;
204 op->flags = 0;
205 op->reserved = nullptr;
206 op++;
207 error = grpc_call_start_batch(c, ops, static_cast<size_t>(op - ops), tag(1),
208 nullptr);
209 GPR_ASSERT(GRPC_CALL_OK == error);
210
211 CQ_EXPECT_COMPLETION(cqv, tag(101), true);
212 CQ_EXPECT_COMPLETION(cqv, tag(1), true);
213 cq_verify(cqv);
214
215 op = ops;
216 op->op = GRPC_OP_SEND_INITIAL_METADATA;
217 op->data.send_initial_metadata.count = 0;
218 op->flags = 0;
219 op->reserved = nullptr;
220 op++;
221 op->op = GRPC_OP_RECV_MESSAGE;
222 op->data.recv_message.recv_message = &request_payload_recv;
223 op->flags = 0;
224 op->reserved = nullptr;
225 op++;
226 error = grpc_call_start_batch(s, ops, static_cast<size_t>(op - ops), tag(102),
227 nullptr);
228 GPR_ASSERT(GRPC_CALL_OK == error);
229
230 CQ_EXPECT_COMPLETION(cqv, tag(102), false);
231
232 op = ops;
233 op->op = GRPC_OP_RECV_CLOSE_ON_SERVER;
234 op->data.recv_close_on_server.cancelled = &was_cancelled;
235 op->flags = 0;
236 op->reserved = nullptr;
237 op++;
238 error = grpc_call_start_batch(s, ops, static_cast<size_t>(op - ops), tag(103),
239 nullptr);
240 GPR_ASSERT(GRPC_CALL_OK == error);
241
242 CQ_EXPECT_COMPLETION(cqv, tag(103), true);
243 cq_verify(cqv);
244
245 /* call was cancelled (closed) ... */
246 GPR_ASSERT(was_cancelled != 0);
247 /* with a certain error */
248 GPR_ASSERT(status == expected_error);
249
250 const char* algo_name = nullptr;
251 GPR_ASSERT(grpc_compression_algorithm_name(algorithm_to_disable, &algo_name));
252 std::string expected_details =
253 absl::StrFormat("Compression algorithm '%s' is disabled.", algo_name);
254 /* and we expect a specific reason for it */
255 GPR_ASSERT(0 == grpc_slice_str_cmp(details, expected_details.c_str()));
256 GPR_ASSERT(0 == grpc_slice_str_cmp(call_details.method, "/foo"));
257
258 grpc_slice_unref(details);
259 grpc_metadata_array_destroy(&initial_metadata_recv);
260 grpc_metadata_array_destroy(&trailing_metadata_recv);
261 grpc_metadata_array_destroy(&request_metadata_recv);
262 grpc_call_details_destroy(&call_details);
263
264 grpc_call_unref(c);
265 grpc_call_unref(s);
266
267 cq_verifier_destroy(cqv);
268
269 grpc_slice_unref(request_payload_slice);
270 grpc_byte_buffer_destroy(request_payload);
271 grpc_byte_buffer_destroy(request_payload_recv);
272 grpc_channel_args_destroy(client_args);
273 grpc_channel_args_destroy(server_args);
274 end_test(&f);
275 config.tear_down_data(&f);
276 }
277
request_with_payload_template_inner(grpc_end2end_test_config config,const char * test_name,uint32_t client_send_flags_bitmask,grpc_compression_algorithm default_client_channel_compression_algorithm,grpc_compression_algorithm default_server_channel_compression_algorithm,grpc_compression_algorithm expected_algorithm_from_client,grpc_compression_algorithm expected_algorithm_from_server,grpc_metadata * client_init_metadata,bool set_server_level,grpc_compression_level server_compression_level,bool send_message_before_initial_metadata,bool decompress_in_core)278 static void request_with_payload_template_inner(
279 grpc_end2end_test_config config, const char* test_name,
280 uint32_t client_send_flags_bitmask,
281 grpc_compression_algorithm default_client_channel_compression_algorithm,
282 grpc_compression_algorithm default_server_channel_compression_algorithm,
283 grpc_compression_algorithm expected_algorithm_from_client,
284 grpc_compression_algorithm expected_algorithm_from_server,
285 grpc_metadata* client_init_metadata, bool set_server_level,
286 grpc_compression_level server_compression_level,
287 bool send_message_before_initial_metadata, bool decompress_in_core) {
288 grpc_call* c;
289 grpc_call* s;
290 grpc_slice request_payload_slice;
291 grpc_byte_buffer* request_payload = nullptr;
292 grpc_channel_args* client_args;
293 grpc_channel_args* server_args;
294 grpc_end2end_test_fixture f;
295 grpc_op ops[6];
296 grpc_op* op;
297 grpc_metadata_array initial_metadata_recv;
298 grpc_metadata_array trailing_metadata_recv;
299 grpc_metadata_array request_metadata_recv;
300 grpc_byte_buffer* request_payload_recv = nullptr;
301 grpc_byte_buffer* response_payload;
302 grpc_byte_buffer* response_payload_recv;
303 grpc_call_details call_details;
304 grpc_status_code status;
305 grpc_call_error error;
306 grpc_slice details;
307 int was_cancelled = 2;
308 cq_verifier* cqv;
309 char request_str[1024];
310 char response_str[1024];
311
312 memset(request_str, 'x', 1023);
313 request_str[1023] = '\0';
314
315 memset(response_str, 'y', 1023);
316 response_str[1023] = '\0';
317
318 request_payload_slice = grpc_slice_from_copied_string(request_str);
319 grpc_slice response_payload_slice =
320 grpc_slice_from_copied_string(response_str);
321
322 client_args = grpc_channel_args_set_channel_default_compression_algorithm(
323 nullptr, default_client_channel_compression_algorithm);
324 server_args = grpc_channel_args_set_channel_default_compression_algorithm(
325 nullptr, default_server_channel_compression_algorithm);
326 if (!decompress_in_core) {
327 grpc_arg disable_decompression_in_core_arg =
328 grpc_channel_arg_integer_create(
329 const_cast<char*>(GRPC_ARG_ENABLE_PER_MESSAGE_DECOMPRESSION), 0);
330 grpc_channel_args* old_client_args = client_args;
331 grpc_channel_args* old_server_args = server_args;
332 client_args = grpc_channel_args_copy_and_add(
333 client_args, &disable_decompression_in_core_arg, 1);
334 server_args = grpc_channel_args_copy_and_add(
335 server_args, &disable_decompression_in_core_arg, 1);
336 grpc_channel_args_destroy(old_client_args);
337 grpc_channel_args_destroy(old_server_args);
338 }
339 f = begin_test(config, test_name, client_args, server_args,
340 decompress_in_core);
341 cqv = cq_verifier_create(f.cq);
342
343 gpr_timespec deadline = five_seconds_from_now();
344 c = grpc_channel_create_call(f.client, nullptr, GRPC_PROPAGATE_DEFAULTS, f.cq,
345 grpc_slice_from_static_string("/foo"), nullptr,
346 deadline, nullptr);
347 GPR_ASSERT(c);
348
349 grpc_metadata_array_init(&initial_metadata_recv);
350 grpc_metadata_array_init(&trailing_metadata_recv);
351 grpc_metadata_array_init(&request_metadata_recv);
352 grpc_call_details_init(&call_details);
353
354 if (send_message_before_initial_metadata) {
355 request_payload = grpc_raw_byte_buffer_create(&request_payload_slice, 1);
356 memset(ops, 0, sizeof(ops));
357 op = ops;
358 op->op = GRPC_OP_SEND_MESSAGE;
359 op->data.send_message.send_message = request_payload;
360 op->flags = client_send_flags_bitmask;
361 op->reserved = nullptr;
362 op++;
363 error = grpc_call_start_batch(c, ops, static_cast<size_t>(op - ops), tag(2),
364 nullptr);
365 GPR_ASSERT(GRPC_CALL_OK == error);
366 CQ_EXPECT_COMPLETION(cqv, tag(2), true);
367 }
368 memset(ops, 0, sizeof(ops));
369 op = ops;
370 op->op = GRPC_OP_SEND_INITIAL_METADATA;
371 if (client_init_metadata != nullptr) {
372 op->data.send_initial_metadata.count = 1;
373 op->data.send_initial_metadata.metadata = client_init_metadata;
374 } else {
375 op->data.send_initial_metadata.count = 0;
376 }
377 op->flags = 0;
378 op->reserved = nullptr;
379 op++;
380 op->op = GRPC_OP_RECV_INITIAL_METADATA;
381 op->data.recv_initial_metadata.recv_initial_metadata = &initial_metadata_recv;
382 op->flags = 0;
383 op->reserved = nullptr;
384 op++;
385 op->op = GRPC_OP_RECV_STATUS_ON_CLIENT;
386 op->data.recv_status_on_client.trailing_metadata = &trailing_metadata_recv;
387 op->data.recv_status_on_client.status = &status;
388 op->data.recv_status_on_client.status_details = &details;
389 op->flags = 0;
390 op->reserved = nullptr;
391 op++;
392 error = grpc_call_start_batch(c, ops, static_cast<size_t>(op - ops), tag(1),
393 nullptr);
394 GPR_ASSERT(GRPC_CALL_OK == error);
395
396 error =
397 grpc_server_request_call(f.server, &s, &call_details,
398 &request_metadata_recv, f.cq, f.cq, tag(100));
399 GPR_ASSERT(GRPC_CALL_OK == error);
400 CQ_EXPECT_COMPLETION(cqv, tag(100), true);
401 cq_verify(cqv);
402
403 GPR_ASSERT(GPR_BITCOUNT(grpc_call_test_only_get_encodings_accepted_by_peer(
404 s)) == GRPC_COMPRESS_ALGORITHMS_COUNT);
405 GPR_ASSERT(GPR_BITGET(grpc_call_test_only_get_encodings_accepted_by_peer(s),
406 GRPC_COMPRESS_NONE) != 0);
407 GPR_ASSERT(GPR_BITGET(grpc_call_test_only_get_encodings_accepted_by_peer(s),
408 GRPC_COMPRESS_DEFLATE) != 0);
409 GPR_ASSERT(GPR_BITGET(grpc_call_test_only_get_encodings_accepted_by_peer(s),
410 GRPC_COMPRESS_GZIP) != 0);
411 memset(ops, 0, sizeof(ops));
412 op = ops;
413 op->op = GRPC_OP_SEND_INITIAL_METADATA;
414 op->data.send_initial_metadata.count = 0;
415 if (set_server_level) {
416 op->data.send_initial_metadata.maybe_compression_level.is_set = true;
417 op->data.send_initial_metadata.maybe_compression_level.level =
418 server_compression_level;
419 }
420 op->flags = 0;
421 op->reserved = nullptr;
422 op++;
423 op->op = GRPC_OP_RECV_CLOSE_ON_SERVER;
424 op->data.recv_close_on_server.cancelled = &was_cancelled;
425 op->flags = 0;
426 op->reserved = nullptr;
427 op++;
428 error = grpc_call_start_batch(s, ops, static_cast<size_t>(op - ops), tag(101),
429 nullptr);
430 GPR_ASSERT(GRPC_CALL_OK == error);
431 for (int i = 0; i < 2; i++) {
432 response_payload = grpc_raw_byte_buffer_create(&response_payload_slice, 1);
433
434 if (i > 0 || !send_message_before_initial_metadata) {
435 request_payload = grpc_raw_byte_buffer_create(&request_payload_slice, 1);
436 memset(ops, 0, sizeof(ops));
437 op = ops;
438 op->op = GRPC_OP_SEND_MESSAGE;
439 op->data.send_message.send_message = request_payload;
440 op->flags = client_send_flags_bitmask;
441 op->reserved = nullptr;
442 op++;
443 error = grpc_call_start_batch(c, ops, static_cast<size_t>(op - ops),
444 tag(2), nullptr);
445 GPR_ASSERT(GRPC_CALL_OK == error);
446 CQ_EXPECT_COMPLETION(cqv, tag(2), 1);
447 }
448
449 memset(ops, 0, sizeof(ops));
450 op = ops;
451 op->op = GRPC_OP_RECV_MESSAGE;
452 op->data.recv_message.recv_message = &request_payload_recv;
453 op->flags = 0;
454 op->reserved = nullptr;
455 op++;
456 error = grpc_call_start_batch(s, ops, static_cast<size_t>(op - ops),
457 tag(102), nullptr);
458 GPR_ASSERT(GRPC_CALL_OK == error);
459
460 CQ_EXPECT_COMPLETION(cqv, tag(102), 1);
461 cq_verify(cqv);
462
463 GPR_ASSERT(request_payload_recv->type == GRPC_BB_RAW);
464 GPR_ASSERT(byte_buffer_eq_string(request_payload_recv, request_str));
465 GPR_ASSERT(request_payload_recv->data.raw.compression ==
466 (decompress_in_core ? GRPC_COMPRESS_NONE
467 : expected_algorithm_from_client));
468
469 memset(ops, 0, sizeof(ops));
470 op = ops;
471 op->op = GRPC_OP_SEND_MESSAGE;
472 op->data.send_message.send_message = response_payload;
473 op->flags = 0;
474 op->reserved = nullptr;
475 op++;
476 error = grpc_call_start_batch(s, ops, static_cast<size_t>(op - ops),
477 tag(103), nullptr);
478 GPR_ASSERT(GRPC_CALL_OK == error);
479
480 memset(ops, 0, sizeof(ops));
481 op = ops;
482 op->op = GRPC_OP_RECV_MESSAGE;
483 op->data.recv_message.recv_message = &response_payload_recv;
484 op->flags = 0;
485 op->reserved = nullptr;
486 op++;
487 error = grpc_call_start_batch(c, ops, static_cast<size_t>(op - ops), tag(3),
488 nullptr);
489 GPR_ASSERT(GRPC_CALL_OK == error);
490
491 CQ_EXPECT_COMPLETION(cqv, tag(103), 1);
492 CQ_EXPECT_COMPLETION(cqv, tag(3), 1);
493 cq_verify(cqv);
494
495 GPR_ASSERT(response_payload_recv->type == GRPC_BB_RAW);
496 GPR_ASSERT(byte_buffer_eq_string(response_payload_recv, response_str));
497 if (server_compression_level > GRPC_COMPRESS_LEVEL_NONE) {
498 const grpc_compression_algorithm algo_for_server_level =
499 grpc_call_compression_for_level(s, server_compression_level);
500 GPR_ASSERT(
501 response_payload_recv->data.raw.compression ==
502 (decompress_in_core ? GRPC_COMPRESS_NONE : algo_for_server_level));
503 } else {
504 GPR_ASSERT(response_payload_recv->data.raw.compression ==
505 (decompress_in_core ? GRPC_COMPRESS_NONE
506 : expected_algorithm_from_server));
507 }
508
509 grpc_byte_buffer_destroy(request_payload);
510 grpc_byte_buffer_destroy(response_payload);
511 grpc_byte_buffer_destroy(request_payload_recv);
512 grpc_byte_buffer_destroy(response_payload_recv);
513 }
514 grpc_slice_unref(request_payload_slice);
515 grpc_slice_unref(response_payload_slice);
516
517 memset(ops, 0, sizeof(ops));
518 op = ops;
519 op->op = GRPC_OP_SEND_CLOSE_FROM_CLIENT;
520 op->flags = 0;
521 op->reserved = nullptr;
522 op++;
523 error = grpc_call_start_batch(c, ops, static_cast<size_t>(op - ops), tag(4),
524 nullptr);
525 GPR_ASSERT(GRPC_CALL_OK == error);
526
527 memset(ops, 0, sizeof(ops));
528 op = ops;
529 op->op = GRPC_OP_SEND_STATUS_FROM_SERVER;
530 op->data.send_status_from_server.trailing_metadata_count = 0;
531 op->data.send_status_from_server.status = GRPC_STATUS_OK;
532 grpc_slice status_details = grpc_slice_from_static_string("xyz");
533 op->data.send_status_from_server.status_details = &status_details;
534 op->flags = 0;
535 op->reserved = nullptr;
536 op++;
537 error = grpc_call_start_batch(s, ops, static_cast<size_t>(op - ops), tag(104),
538 nullptr);
539 GPR_ASSERT(GRPC_CALL_OK == error);
540
541 CQ_EXPECT_COMPLETION(cqv, tag(1), 1);
542 CQ_EXPECT_COMPLETION(cqv, tag(4), 1);
543 CQ_EXPECT_COMPLETION(cqv, tag(101), 1);
544 CQ_EXPECT_COMPLETION(cqv, tag(104), 1);
545 cq_verify(cqv);
546
547 GPR_ASSERT(status == GRPC_STATUS_OK);
548 GPR_ASSERT(0 == grpc_slice_str_cmp(details, "xyz"));
549 GPR_ASSERT(0 == grpc_slice_str_cmp(call_details.method, "/foo"));
550 GPR_ASSERT(was_cancelled == 0);
551
552 grpc_slice_unref(details);
553 grpc_metadata_array_destroy(&initial_metadata_recv);
554 grpc_metadata_array_destroy(&trailing_metadata_recv);
555 grpc_metadata_array_destroy(&request_metadata_recv);
556 grpc_call_details_destroy(&call_details);
557
558 grpc_call_unref(c);
559 grpc_call_unref(s);
560
561 cq_verifier_destroy(cqv);
562 grpc_channel_args_destroy(client_args);
563 grpc_channel_args_destroy(server_args);
564 end_test(&f);
565 config.tear_down_data(&f);
566 }
567
request_with_payload_template(grpc_end2end_test_config config,const char * test_name,uint32_t client_send_flags_bitmask,grpc_compression_algorithm default_client_channel_compression_algorithm,grpc_compression_algorithm default_server_channel_compression_algorithm,grpc_compression_algorithm expected_algorithm_from_client,grpc_compression_algorithm expected_algorithm_from_server,grpc_metadata * client_init_metadata,bool set_server_level,grpc_compression_level server_compression_level,bool send_message_before_initial_metadata)568 static void request_with_payload_template(
569 grpc_end2end_test_config config, const char* test_name,
570 uint32_t client_send_flags_bitmask,
571 grpc_compression_algorithm default_client_channel_compression_algorithm,
572 grpc_compression_algorithm default_server_channel_compression_algorithm,
573 grpc_compression_algorithm expected_algorithm_from_client,
574 grpc_compression_algorithm expected_algorithm_from_server,
575 grpc_metadata* client_init_metadata, bool set_server_level,
576 grpc_compression_level server_compression_level,
577 bool send_message_before_initial_metadata) {
578 request_with_payload_template_inner(
579 config, test_name, client_send_flags_bitmask,
580 default_client_channel_compression_algorithm,
581 default_server_channel_compression_algorithm,
582 expected_algorithm_from_client, expected_algorithm_from_server,
583 client_init_metadata, set_server_level, server_compression_level,
584 send_message_before_initial_metadata, false);
585 request_with_payload_template_inner(
586 config, test_name, client_send_flags_bitmask,
587 default_client_channel_compression_algorithm,
588 default_server_channel_compression_algorithm,
589 expected_algorithm_from_client, expected_algorithm_from_server,
590 client_init_metadata, set_server_level, server_compression_level,
591 send_message_before_initial_metadata, true);
592 }
593
test_invoke_request_with_exceptionally_uncompressed_payload(grpc_end2end_test_config config)594 static void test_invoke_request_with_exceptionally_uncompressed_payload(
595 grpc_end2end_test_config config) {
596 request_with_payload_template(
597 config, "test_invoke_request_with_exceptionally_uncompressed_payload",
598 GRPC_WRITE_NO_COMPRESS, GRPC_COMPRESS_GZIP, GRPC_COMPRESS_GZIP,
599 GRPC_COMPRESS_NONE, GRPC_COMPRESS_GZIP, nullptr, false,
600 /* ignored */ GRPC_COMPRESS_LEVEL_NONE, false);
601 }
602
test_invoke_request_with_uncompressed_payload(grpc_end2end_test_config config)603 static void test_invoke_request_with_uncompressed_payload(
604 grpc_end2end_test_config config) {
605 request_with_payload_template(
606 config, "test_invoke_request_with_uncompressed_payload", 0,
607 GRPC_COMPRESS_NONE, GRPC_COMPRESS_NONE, GRPC_COMPRESS_NONE,
608 GRPC_COMPRESS_NONE, nullptr, false,
609 /* ignored */ GRPC_COMPRESS_LEVEL_NONE, false);
610 }
611
test_invoke_request_with_compressed_payload(grpc_end2end_test_config config)612 static void test_invoke_request_with_compressed_payload(
613 grpc_end2end_test_config config) {
614 request_with_payload_template(
615 config, "test_invoke_request_with_compressed_payload", 0,
616 GRPC_COMPRESS_GZIP, GRPC_COMPRESS_GZIP, GRPC_COMPRESS_GZIP,
617 GRPC_COMPRESS_GZIP, nullptr, false,
618 /* ignored */ GRPC_COMPRESS_LEVEL_NONE, false);
619 }
620
test_invoke_request_with_send_message_before_initial_metadata(grpc_end2end_test_config config)621 static void test_invoke_request_with_send_message_before_initial_metadata(
622 grpc_end2end_test_config config) {
623 request_with_payload_template(
624 config, "test_invoke_request_with_compressed_payload", 0,
625 GRPC_COMPRESS_GZIP, GRPC_COMPRESS_GZIP, GRPC_COMPRESS_GZIP,
626 GRPC_COMPRESS_GZIP, nullptr, false,
627 /* ignored */ GRPC_COMPRESS_LEVEL_NONE, true);
628 }
629
test_invoke_request_with_server_level(grpc_end2end_test_config config)630 static void test_invoke_request_with_server_level(
631 grpc_end2end_test_config config) {
632 request_with_payload_template(
633 config, "test_invoke_request_with_server_level", 0, GRPC_COMPRESS_NONE,
634 GRPC_COMPRESS_NONE, GRPC_COMPRESS_NONE, GRPC_COMPRESS_NONE /* ignored */,
635 nullptr, true, GRPC_COMPRESS_LEVEL_HIGH, false);
636 }
637
test_invoke_request_with_compressed_payload_md_override(grpc_end2end_test_config config)638 static void test_invoke_request_with_compressed_payload_md_override(
639 grpc_end2end_test_config config) {
640 grpc_metadata gzip_compression_override;
641 grpc_metadata identity_compression_override;
642
643 gzip_compression_override.key = GRPC_MDSTR_GRPC_INTERNAL_ENCODING_REQUEST;
644 gzip_compression_override.value = grpc_slice_from_static_string("gzip");
645 memset(&gzip_compression_override.internal_data, 0,
646 sizeof(gzip_compression_override.internal_data));
647
648 identity_compression_override.key = GRPC_MDSTR_GRPC_INTERNAL_ENCODING_REQUEST;
649 identity_compression_override.value =
650 grpc_slice_from_static_string("identity");
651 memset(&identity_compression_override.internal_data, 0,
652 sizeof(identity_compression_override.internal_data));
653
654 /* Channel default NONE (aka IDENTITY), call override to GZIP */
655 request_with_payload_template(
656 config, "test_invoke_request_with_compressed_payload_md_override_1", 0,
657 GRPC_COMPRESS_NONE, GRPC_COMPRESS_NONE, GRPC_COMPRESS_GZIP,
658 GRPC_COMPRESS_NONE, &gzip_compression_override, false,
659 /*ignored*/ GRPC_COMPRESS_LEVEL_NONE, false);
660
661 /* Channel default DEFLATE, call override to GZIP */
662 request_with_payload_template(
663 config, "test_invoke_request_with_compressed_payload_md_override_2", 0,
664 GRPC_COMPRESS_DEFLATE, GRPC_COMPRESS_NONE, GRPC_COMPRESS_GZIP,
665 GRPC_COMPRESS_NONE, &gzip_compression_override, false,
666 /*ignored*/ GRPC_COMPRESS_LEVEL_NONE, false);
667
668 /* Channel default DEFLATE, call override to NONE (aka IDENTITY) */
669 request_with_payload_template(
670 config, "test_invoke_request_with_compressed_payload_md_override_3", 0,
671 GRPC_COMPRESS_DEFLATE, GRPC_COMPRESS_NONE, GRPC_COMPRESS_NONE,
672 GRPC_COMPRESS_NONE, &identity_compression_override, false,
673 /*ignored*/ GRPC_COMPRESS_LEVEL_NONE, false);
674 }
675
test_invoke_request_with_disabled_algorithm(grpc_end2end_test_config config)676 static void test_invoke_request_with_disabled_algorithm(
677 grpc_end2end_test_config config) {
678 request_for_disabled_algorithm(config,
679 "test_invoke_request_with_disabled_algorithm",
680 0, GRPC_COMPRESS_GZIP, GRPC_COMPRESS_GZIP,
681 GRPC_STATUS_UNIMPLEMENTED, nullptr, false);
682 request_for_disabled_algorithm(config,
683 "test_invoke_request_with_disabled_algorithm",
684 0, GRPC_COMPRESS_GZIP, GRPC_COMPRESS_GZIP,
685 GRPC_STATUS_UNIMPLEMENTED, nullptr, true);
686 }
687
compressed_payload(grpc_end2end_test_config config)688 void compressed_payload(grpc_end2end_test_config config) {
689 test_invoke_request_with_exceptionally_uncompressed_payload(config);
690 test_invoke_request_with_uncompressed_payload(config);
691 test_invoke_request_with_compressed_payload(config);
692 test_invoke_request_with_send_message_before_initial_metadata(config);
693 test_invoke_request_with_server_level(config);
694 test_invoke_request_with_compressed_payload_md_override(config);
695 test_invoke_request_with_disabled_algorithm(config);
696 }
697
compressed_payload_pre_init(void)698 void compressed_payload_pre_init(void) {}
699