1# Copyright 2015 gRPC authors.
2#
3# Licensed under the Apache License, Version 2.0 (the "License");
4# you may not use this file except in compliance with the License.
5# You may obtain a copy of the License at
6#
7#     http://www.apache.org/licenses/LICENSE-2.0
8#
9# Unless required by applicable law or agreed to in writing, software
10# distributed under the License is distributed on an "AS IS" BASIS,
11# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12# See the License for the specific language governing permissions and
13# limitations under the License.
14"""Common code used throughout tests of gRPC."""
15
16import collections
17
18from concurrent import futures
19import grpc
20import six
21
22INVOCATION_INITIAL_METADATA = (
23    ('0', 'abc'),
24    ('1', 'def'),
25    ('2', 'ghi'),
26)
27SERVICE_INITIAL_METADATA = (
28    ('3', 'jkl'),
29    ('4', 'mno'),
30    ('5', 'pqr'),
31)
32SERVICE_TERMINAL_METADATA = (
33    ('6', 'stu'),
34    ('7', 'vwx'),
35    ('8', 'yza'),
36)
37DETAILS = 'test details'
38
39
40def metadata_transmitted(original_metadata, transmitted_metadata):
41    """Judges whether or not metadata was acceptably transmitted.
42
43  gRPC is allowed to insert key-value pairs into the metadata values given by
44  applications and to reorder key-value pairs with different keys but it is not
45  allowed to alter existing key-value pairs or to reorder key-value pairs with
46  the same key.
47
48  Args:
49    original_metadata: A metadata value used in a test of gRPC. An iterable over
50      iterables of length 2.
51    transmitted_metadata: A metadata value corresponding to original_metadata
52      after having been transmitted via gRPC. An iterable over iterables of
53      length 2.
54
55  Returns:
56     A boolean indicating whether transmitted_metadata accurately reflects
57      original_metadata after having been transmitted via gRPC.
58  """
59    original = collections.defaultdict(list)
60    for key, value in original_metadata:
61        original[key].append(value)
62    transmitted = collections.defaultdict(list)
63    for key, value in transmitted_metadata:
64        transmitted[key].append(value)
65
66    for key, values in six.iteritems(original):
67        transmitted_values = transmitted[key]
68        transmitted_iterator = iter(transmitted_values)
69        try:
70            for value in values:
71                while True:
72                    transmitted_value = next(transmitted_iterator)
73                    if value == transmitted_value:
74                        break
75        except StopIteration:
76            return False
77    else:
78        return True
79
80
81def test_secure_channel(target, channel_credentials, server_host_override):
82    """Creates an insecure Channel to a remote host.
83
84  Args:
85    host: The name of the remote host to which to connect.
86    port: The port of the remote host to which to connect.
87    channel_credentials: The implementations.ChannelCredentials with which to
88      connect.
89    server_host_override: The target name used for SSL host name checking.
90
91  Returns:
92    An implementations.Channel to the remote host through which RPCs may be
93      conducted.
94  """
95    channel = grpc.secure_channel(target, channel_credentials, ((
96        'grpc.ssl_target_name_override',
97        server_host_override,
98    ),))
99    return channel
100
101
102def test_server(max_workers=10):
103    """Creates an insecure grpc server.
104
105     These servers have SO_REUSEPORT disabled to prevent cross-talk.
106     """
107    return grpc.server(
108        futures.ThreadPoolExecutor(max_workers=max_workers),
109        options=(('grpc.so_reuseport', 0),))
110