1# Copyright 2018 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
15cimport cpython
16
17
18# TODO(https://github.com/grpc/grpc/issues/15662): Reform this.
19cdef void* _copy_pointer(void* pointer):
20  return pointer
21
22
23# TODO(https://github.com/grpc/grpc/issues/15662): Reform this.
24cdef void _destroy_pointer(void* pointer):
25  pass
26
27
28cdef int _compare_pointer(void* first_pointer, void* second_pointer):
29  if first_pointer < second_pointer:
30    return -1
31  elif first_pointer > second_pointer:
32    return 1
33  else:
34    return 0
35
36
37cdef class _GrpcArgWrapper:
38
39  cdef grpc_arg arg
40
41
42cdef tuple _wrap_grpc_arg(grpc_arg arg):
43  wrapped = _GrpcArgWrapper()
44  wrapped.arg = arg
45  return ("grpc.python._cygrpc._GrpcArgWrapper", wrapped)
46
47
48cdef grpc_arg _unwrap_grpc_arg(tuple wrapped_arg):
49  cdef _GrpcArgWrapper wrapped = wrapped_arg[1]
50  return wrapped.arg
51
52
53cdef class _ArgumentProcessor:
54
55  cdef void c(self, argument, grpc_arg_pointer_vtable *vtable, references):
56    key, value = argument
57    cdef bytes encoded_key = _encode(key)
58    if encoded_key is not key:
59      references.append(encoded_key)
60    self.c_argument.key = encoded_key
61    if isinstance(value, int):
62      self.c_argument.type = GRPC_ARG_INTEGER
63      self.c_argument.value.integer = value
64    elif isinstance(value, (bytes, str, unicode,)):
65      self.c_argument.type = GRPC_ARG_STRING
66      encoded_value = _encode(value)
67      if encoded_value is not value:
68        references.append(encoded_value)
69      self.c_argument.value.string = encoded_value
70    elif isinstance(value, _GrpcArgWrapper):
71      self.c_argument = (<_GrpcArgWrapper>value).arg
72    elif hasattr(value, '__int__'):
73      # Pointer objects must override __int__() to return
74      # the underlying C address (Python ints are word size). The
75      # lifecycle of the pointer is fixed to the lifecycle of the
76      # python object wrapping it.
77      self.c_argument.type = GRPC_ARG_POINTER
78      self.c_argument.value.pointer.vtable = vtable
79      self.c_argument.value.pointer.address = <void*>(<intptr_t>int(value))
80    else:
81      raise TypeError(
82          'Expected int, bytes, or behavior, got {}'.format(type(value)))
83
84
85cdef class _ArgumentsProcessor:
86
87  def __cinit__(self, arguments):
88    self._arguments = () if arguments is None else tuple(arguments)
89    self._argument_processors = []
90    self._references = []
91
92  cdef grpc_channel_args *c(self, grpc_arg_pointer_vtable *vtable):
93    self._c_arguments.arguments_length = len(self._arguments)
94    if self._c_arguments.arguments_length == 0:
95      return NULL
96    else:
97      self._c_arguments.arguments = <grpc_arg *>gpr_malloc(
98          self._c_arguments.arguments_length * sizeof(grpc_arg))
99      for index, argument in enumerate(self._arguments):
100        argument_processor = _ArgumentProcessor()
101        argument_processor.c(argument, vtable, self._references)
102        self._c_arguments.arguments[index] = argument_processor.c_argument
103        self._argument_processors.append(argument_processor)
104      return &self._c_arguments
105
106  cdef un_c(self):
107    if self._arguments:
108      gpr_free(self._c_arguments.arguments)
109