• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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