|  | @@ -0,0 +1,88 @@
 | 
	
		
			
				|  |  | +# Copyright 2018 gRPC authors.
 | 
	
		
			
				|  |  | +#
 | 
	
		
			
				|  |  | +# Licensed under the Apache License, Version 2.0 (the "License");
 | 
	
		
			
				|  |  | +# you may not use this file except in compliance with the License.
 | 
	
		
			
				|  |  | +# You may obtain a copy of the License at
 | 
	
		
			
				|  |  | +#
 | 
	
		
			
				|  |  | +#     http://www.apache.org/licenses/LICENSE-2.0
 | 
	
		
			
				|  |  | +#
 | 
	
		
			
				|  |  | +# Unless required by applicable law or agreed to in writing, software
 | 
	
		
			
				|  |  | +# distributed under the License is distributed on an "AS IS" BASIS,
 | 
	
		
			
				|  |  | +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
	
		
			
				|  |  | +# See the License for the specific language governing permissions and
 | 
	
		
			
				|  |  | +# limitations under the License.
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +cimport cpython
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +cdef void* _copy_pointer(void* pointer):
 | 
	
		
			
				|  |  | +  return pointer
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +cdef void _destroy_pointer(void* pointer):
 | 
	
		
			
				|  |  | +  pass
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +cdef int _compare_pointer(void* first_pointer, void* second_pointer):
 | 
	
		
			
				|  |  | +  if first_pointer < second_pointer:
 | 
	
		
			
				|  |  | +    return -1
 | 
	
		
			
				|  |  | +  elif first_pointer > second_pointer:
 | 
	
		
			
				|  |  | +    return 1
 | 
	
		
			
				|  |  | +  else:
 | 
	
		
			
				|  |  | +    return 0
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +cdef class _ArgumentProcessor:
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  cdef void c(self, argument, grpc_arg_pointer_vtable *vtable, references):
 | 
	
		
			
				|  |  | +    key, value = argument
 | 
	
		
			
				|  |  | +    cdef bytes encoded_key = _encode(key)
 | 
	
		
			
				|  |  | +    if encoded_key is not key:
 | 
	
		
			
				|  |  | +      references.append(encoded_key)
 | 
	
		
			
				|  |  | +    self.c_argument.key = encoded_key
 | 
	
		
			
				|  |  | +    if isinstance(value, int):
 | 
	
		
			
				|  |  | +      self.c_argument.type = GRPC_ARG_INTEGER
 | 
	
		
			
				|  |  | +      self.c_argument.value.integer = value
 | 
	
		
			
				|  |  | +    elif isinstance(value, (bytes, str, unicode,)):
 | 
	
		
			
				|  |  | +      self.c_argument.type = GRPC_ARG_STRING
 | 
	
		
			
				|  |  | +      encoded_value = _encode(value)
 | 
	
		
			
				|  |  | +      if encoded_value is not value:
 | 
	
		
			
				|  |  | +        references.append(encoded_value)
 | 
	
		
			
				|  |  | +      self.c_argument.value.string = encoded_value
 | 
	
		
			
				|  |  | +    elif hasattr(value, '__int__'):
 | 
	
		
			
				|  |  | +      # Pointer objects must override __int__() to return
 | 
	
		
			
				|  |  | +      # the underlying C address (Python ints are word size). The
 | 
	
		
			
				|  |  | +      # lifecycle of the pointer is fixed to the lifecycle of the
 | 
	
		
			
				|  |  | +      # python object wrapping it.
 | 
	
		
			
				|  |  | +      self.c_argument.type = GRPC_ARG_POINTER
 | 
	
		
			
				|  |  | +      self.c_argument.value.pointer.vtable = vtable
 | 
	
		
			
				|  |  | +      self.c_argument.value.pointer.address = <void*>(<intptr_t>int(value))
 | 
	
		
			
				|  |  | +    else:
 | 
	
		
			
				|  |  | +      raise TypeError(
 | 
	
		
			
				|  |  | +          'Expected int, bytes, or behavior, got {}'.format(type(value)))
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +cdef class _ArgumentsProcessor:
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  def __cinit__(self, arguments):
 | 
	
		
			
				|  |  | +    self._arguments = () if arguments is None else tuple(arguments)
 | 
	
		
			
				|  |  | +    self._argument_processors = []
 | 
	
		
			
				|  |  | +    self._references = []
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  cdef grpc_channel_args *c(self, grpc_arg_pointer_vtable *vtable):
 | 
	
		
			
				|  |  | +    self._c_arguments.arguments_length = len(self._arguments)
 | 
	
		
			
				|  |  | +    if self._c_arguments.arguments_length == 0:
 | 
	
		
			
				|  |  | +      return NULL
 | 
	
		
			
				|  |  | +    else:
 | 
	
		
			
				|  |  | +      self._c_arguments.arguments = <grpc_arg *>gpr_malloc(
 | 
	
		
			
				|  |  | +          self._c_arguments.arguments_length * sizeof(grpc_arg))
 | 
	
		
			
				|  |  | +      for index, argument in enumerate(self._arguments):
 | 
	
		
			
				|  |  | +        argument_processor = _ArgumentProcessor()
 | 
	
		
			
				|  |  | +        argument_processor.c(argument, vtable, self._references)
 | 
	
		
			
				|  |  | +        self._c_arguments.arguments[index] = argument_processor.c_argument
 | 
	
		
			
				|  |  | +        self._argument_processors.append(argument_processor)
 | 
	
		
			
				|  |  | +      return &self._c_arguments
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  cdef un_c(self):
 | 
	
		
			
				|  |  | +    if self._arguments:
 | 
	
		
			
				|  |  | +      gpr_free(self._c_arguments.arguments)
 |