|
|
@ -12,16 +12,14 @@ |
|
|
|
# See the License for the specific language governing permissions and |
|
|
|
# See the License for the specific language governing permissions and |
|
|
|
# limitations under the License. |
|
|
|
# limitations under the License. |
|
|
|
"""Invocation-side implementation of gRPC Asyncio Python.""" |
|
|
|
"""Invocation-side implementation of gRPC Asyncio Python.""" |
|
|
|
|
|
|
|
|
|
|
|
import asyncio |
|
|
|
import asyncio |
|
|
|
from typing import Any, AsyncIterable, Optional, Sequence, AbstractSet |
|
|
|
import sys |
|
|
|
from weakref import WeakSet |
|
|
|
from typing import Any, AsyncIterable, Iterable, Optional, Sequence |
|
|
|
|
|
|
|
|
|
|
|
import logging |
|
|
|
|
|
|
|
import grpc |
|
|
|
import grpc |
|
|
|
from grpc import _common |
|
|
|
from grpc import _common, _compression, _grpcio_metadata |
|
|
|
from grpc._cython import cygrpc |
|
|
|
from grpc._cython import cygrpc |
|
|
|
from grpc import _compression |
|
|
|
|
|
|
|
from grpc import _grpcio_metadata |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
from . import _base_call |
|
|
|
from . import _base_call |
|
|
|
from ._call import (StreamStreamCall, StreamUnaryCall, UnaryStreamCall, |
|
|
|
from ._call import (StreamStreamCall, StreamUnaryCall, UnaryStreamCall, |
|
|
@ -35,6 +33,15 @@ from ._utils import _timeout_to_deadline |
|
|
|
_IMMUTABLE_EMPTY_TUPLE = tuple() |
|
|
|
_IMMUTABLE_EMPTY_TUPLE = tuple() |
|
|
|
_USER_AGENT = 'grpc-python-asyncio/{}'.format(_grpcio_metadata.__version__) |
|
|
|
_USER_AGENT = 'grpc-python-asyncio/{}'.format(_grpcio_metadata.__version__) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if sys.version_info[1] < 7: |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _all_tasks() -> Iterable[asyncio.Task]: |
|
|
|
|
|
|
|
return asyncio.Task.all_tasks() |
|
|
|
|
|
|
|
else: |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _all_tasks() -> Iterable[asyncio.Task]: |
|
|
|
|
|
|
|
return asyncio.all_tasks() |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _augment_channel_arguments(base_options: ChannelArgumentType, |
|
|
|
def _augment_channel_arguments(base_options: ChannelArgumentType, |
|
|
|
compression: Optional[grpc.Compression]): |
|
|
|
compression: Optional[grpc.Compression]): |
|
|
@ -48,50 +55,12 @@ def _augment_channel_arguments(base_options: ChannelArgumentType, |
|
|
|
) + compression_channel_argument + user_agent_channel_argument |
|
|
|
) + compression_channel_argument + user_agent_channel_argument |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class _OngoingCalls: |
|
|
|
|
|
|
|
"""Internal class used for have visibility of the ongoing calls.""" |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
_calls: AbstractSet[_base_call.RpcContext] |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def __init__(self): |
|
|
|
|
|
|
|
self._calls = WeakSet() |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _remove_call(self, call: _base_call.RpcContext): |
|
|
|
|
|
|
|
try: |
|
|
|
|
|
|
|
self._calls.remove(call) |
|
|
|
|
|
|
|
except KeyError: |
|
|
|
|
|
|
|
pass |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@property |
|
|
|
|
|
|
|
def calls(self) -> AbstractSet[_base_call.RpcContext]: |
|
|
|
|
|
|
|
"""Returns the set of ongoing calls.""" |
|
|
|
|
|
|
|
return self._calls |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def size(self) -> int: |
|
|
|
|
|
|
|
"""Returns the number of ongoing calls.""" |
|
|
|
|
|
|
|
return len(self._calls) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def trace_call(self, call: _base_call.RpcContext): |
|
|
|
|
|
|
|
"""Adds and manages a new ongoing call.""" |
|
|
|
|
|
|
|
self._calls.add(call) |
|
|
|
|
|
|
|
call.add_done_callback(self._remove_call) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class _BaseMultiCallable: |
|
|
|
class _BaseMultiCallable: |
|
|
|
"""Base class of all multi callable objects. |
|
|
|
"""Base class of all multi callable objects. |
|
|
|
|
|
|
|
|
|
|
|
Handles the initialization logic and stores common attributes. |
|
|
|
Handles the initialization logic and stores common attributes. |
|
|
|
""" |
|
|
|
""" |
|
|
|
_loop: asyncio.AbstractEventLoop |
|
|
|
_loop: asyncio.AbstractEventLoop |
|
|
|
_channel: cygrpc.AioChannel |
|
|
|
|
|
|
|
_ongoing_calls: _OngoingCalls |
|
|
|
|
|
|
|
_method: bytes |
|
|
|
|
|
|
|
_request_serializer: SerializingFunction |
|
|
|
|
|
|
|
_response_deserializer: DeserializingFunction |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
_channel: cygrpc.AioChannel |
|
|
|
_channel: cygrpc.AioChannel |
|
|
|
_method: bytes |
|
|
|
_method: bytes |
|
|
|
_request_serializer: SerializingFunction |
|
|
|
_request_serializer: SerializingFunction |
|
|
@ -103,7 +72,6 @@ class _BaseMultiCallable: |
|
|
|
def __init__( |
|
|
|
def __init__( |
|
|
|
self, |
|
|
|
self, |
|
|
|
channel: cygrpc.AioChannel, |
|
|
|
channel: cygrpc.AioChannel, |
|
|
|
ongoing_calls: _OngoingCalls, |
|
|
|
|
|
|
|
method: bytes, |
|
|
|
method: bytes, |
|
|
|
request_serializer: SerializingFunction, |
|
|
|
request_serializer: SerializingFunction, |
|
|
|
response_deserializer: DeserializingFunction, |
|
|
|
response_deserializer: DeserializingFunction, |
|
|
@ -112,7 +80,6 @@ class _BaseMultiCallable: |
|
|
|
) -> None: |
|
|
|
) -> None: |
|
|
|
self._loop = loop |
|
|
|
self._loop = loop |
|
|
|
self._channel = channel |
|
|
|
self._channel = channel |
|
|
|
self._ongoing_calls = ongoing_calls |
|
|
|
|
|
|
|
self._method = method |
|
|
|
self._method = method |
|
|
|
self._request_serializer = request_serializer |
|
|
|
self._request_serializer = request_serializer |
|
|
|
self._response_deserializer = response_deserializer |
|
|
|
self._response_deserializer = response_deserializer |
|
|
@ -170,7 +137,6 @@ class UnaryUnaryMultiCallable(_BaseMultiCallable): |
|
|
|
self._request_serializer, self._response_deserializer, |
|
|
|
self._request_serializer, self._response_deserializer, |
|
|
|
self._loop) |
|
|
|
self._loop) |
|
|
|
|
|
|
|
|
|
|
|
self._ongoing_calls.trace_call(call) |
|
|
|
|
|
|
|
return call |
|
|
|
return call |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@ -213,7 +179,7 @@ class UnaryStreamMultiCallable(_BaseMultiCallable): |
|
|
|
wait_for_ready, self._channel, self._method, |
|
|
|
wait_for_ready, self._channel, self._method, |
|
|
|
self._request_serializer, |
|
|
|
self._request_serializer, |
|
|
|
self._response_deserializer, self._loop) |
|
|
|
self._response_deserializer, self._loop) |
|
|
|
self._ongoing_calls.trace_call(call) |
|
|
|
|
|
|
|
return call |
|
|
|
return call |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@ -260,7 +226,7 @@ class StreamUnaryMultiCallable(_BaseMultiCallable): |
|
|
|
credentials, wait_for_ready, self._channel, |
|
|
|
credentials, wait_for_ready, self._channel, |
|
|
|
self._method, self._request_serializer, |
|
|
|
self._method, self._request_serializer, |
|
|
|
self._response_deserializer, self._loop) |
|
|
|
self._response_deserializer, self._loop) |
|
|
|
self._ongoing_calls.trace_call(call) |
|
|
|
|
|
|
|
return call |
|
|
|
return call |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@ -307,7 +273,7 @@ class StreamStreamMultiCallable(_BaseMultiCallable): |
|
|
|
credentials, wait_for_ready, self._channel, |
|
|
|
credentials, wait_for_ready, self._channel, |
|
|
|
self._method, self._request_serializer, |
|
|
|
self._method, self._request_serializer, |
|
|
|
self._response_deserializer, self._loop) |
|
|
|
self._response_deserializer, self._loop) |
|
|
|
self._ongoing_calls.trace_call(call) |
|
|
|
|
|
|
|
return call |
|
|
|
return call |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@ -319,7 +285,6 @@ class Channel: |
|
|
|
_loop: asyncio.AbstractEventLoop |
|
|
|
_loop: asyncio.AbstractEventLoop |
|
|
|
_channel: cygrpc.AioChannel |
|
|
|
_channel: cygrpc.AioChannel |
|
|
|
_unary_unary_interceptors: Optional[Sequence[UnaryUnaryClientInterceptor]] |
|
|
|
_unary_unary_interceptors: Optional[Sequence[UnaryUnaryClientInterceptor]] |
|
|
|
_ongoing_calls: _OngoingCalls |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def __init__(self, target: str, options: ChannelArgumentType, |
|
|
|
def __init__(self, target: str, options: ChannelArgumentType, |
|
|
|
credentials: Optional[grpc.ChannelCredentials], |
|
|
|
credentials: Optional[grpc.ChannelCredentials], |
|
|
@ -359,7 +324,6 @@ class Channel: |
|
|
|
_common.encode(target), |
|
|
|
_common.encode(target), |
|
|
|
_augment_channel_arguments(options, compression), credentials, |
|
|
|
_augment_channel_arguments(options, compression), credentials, |
|
|
|
self._loop) |
|
|
|
self._loop) |
|
|
|
self._ongoing_calls = _OngoingCalls() |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
async def __aenter__(self): |
|
|
|
async def __aenter__(self): |
|
|
|
"""Starts an asynchronous context manager. |
|
|
|
"""Starts an asynchronous context manager. |
|
|
@ -383,22 +347,48 @@ class Channel: |
|
|
|
# No new calls will be accepted by the Cython channel. |
|
|
|
# No new calls will be accepted by the Cython channel. |
|
|
|
self._channel.closing() |
|
|
|
self._channel.closing() |
|
|
|
|
|
|
|
|
|
|
|
if grace: |
|
|
|
# Iterate through running tasks |
|
|
|
# pylint: disable=unused-variable |
|
|
|
tasks = _all_tasks() |
|
|
|
_, pending = await asyncio.wait(self._ongoing_calls.calls, |
|
|
|
calls = [] |
|
|
|
timeout=grace, |
|
|
|
call_tasks = [] |
|
|
|
loop=self._loop) |
|
|
|
for task in tasks: |
|
|
|
|
|
|
|
stack = task.get_stack(limit=1) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# If the Task is created by a C-extension, the stack will be empty. |
|
|
|
|
|
|
|
if not stack: |
|
|
|
|
|
|
|
continue |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Locate ones created by `aio.Call`. |
|
|
|
|
|
|
|
frame = stack[0] |
|
|
|
|
|
|
|
candidate = frame.f_locals.get('self') |
|
|
|
|
|
|
|
if candidate: |
|
|
|
|
|
|
|
if isinstance(candidate, _base_call.Call): |
|
|
|
|
|
|
|
if hasattr(candidate, '_channel'): |
|
|
|
|
|
|
|
# For intercepted Call object |
|
|
|
|
|
|
|
if candidate._channel is not self._channel: |
|
|
|
|
|
|
|
continue |
|
|
|
|
|
|
|
elif hasattr(candidate, '_cython_call'): |
|
|
|
|
|
|
|
# For normal Call object |
|
|
|
|
|
|
|
if candidate._cython_call._channel is not self._channel: |
|
|
|
|
|
|
|
continue |
|
|
|
|
|
|
|
else: |
|
|
|
|
|
|
|
# Unidentified Call object |
|
|
|
|
|
|
|
raise cygrpc.InternalError( |
|
|
|
|
|
|
|
f'Unrecognized call object: {candidate}') |
|
|
|
|
|
|
|
|
|
|
|
if not pending: |
|
|
|
calls.append(candidate) |
|
|
|
return |
|
|
|
call_tasks.append(task) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# If needed, try to wait for them to finish. |
|
|
|
|
|
|
|
# Call objects are not always awaitables. |
|
|
|
|
|
|
|
if grace and call_tasks: |
|
|
|
|
|
|
|
await asyncio.wait(call_tasks, timeout=grace, loop=self._loop) |
|
|
|
|
|
|
|
|
|
|
|
# A new set is created acting as a shallow copy because |
|
|
|
# Time to cancel existing calls. |
|
|
|
# when cancellation happens the calls are automatically |
|
|
|
|
|
|
|
# removed from the originally set. |
|
|
|
|
|
|
|
calls = WeakSet(data=self._ongoing_calls.calls) |
|
|
|
|
|
|
|
for call in calls: |
|
|
|
for call in calls: |
|
|
|
call.cancel() |
|
|
|
call.cancel() |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Destroy the channel |
|
|
|
self._channel.close() |
|
|
|
self._channel.close() |
|
|
|
|
|
|
|
|
|
|
|
async def close(self, grace: Optional[float] = None): |
|
|
|
async def close(self, grace: Optional[float] = None): |
|
|
@ -487,8 +477,7 @@ class Channel: |
|
|
|
Returns: |
|
|
|
Returns: |
|
|
|
A UnaryUnaryMultiCallable value for the named unary-unary method. |
|
|
|
A UnaryUnaryMultiCallable value for the named unary-unary method. |
|
|
|
""" |
|
|
|
""" |
|
|
|
return UnaryUnaryMultiCallable(self._channel, self._ongoing_calls, |
|
|
|
return UnaryUnaryMultiCallable(self._channel, _common.encode(method), |
|
|
|
_common.encode(method), |
|
|
|
|
|
|
|
request_serializer, |
|
|
|
request_serializer, |
|
|
|
response_deserializer, |
|
|
|
response_deserializer, |
|
|
|
self._unary_unary_interceptors, |
|
|
|
self._unary_unary_interceptors, |
|
|
@ -500,8 +489,7 @@ class Channel: |
|
|
|
request_serializer: Optional[SerializingFunction] = None, |
|
|
|
request_serializer: Optional[SerializingFunction] = None, |
|
|
|
response_deserializer: Optional[DeserializingFunction] = None |
|
|
|
response_deserializer: Optional[DeserializingFunction] = None |
|
|
|
) -> UnaryStreamMultiCallable: |
|
|
|
) -> UnaryStreamMultiCallable: |
|
|
|
return UnaryStreamMultiCallable(self._channel, self._ongoing_calls, |
|
|
|
return UnaryStreamMultiCallable(self._channel, _common.encode(method), |
|
|
|
_common.encode(method), |
|
|
|
|
|
|
|
request_serializer, |
|
|
|
request_serializer, |
|
|
|
response_deserializer, None, self._loop) |
|
|
|
response_deserializer, None, self._loop) |
|
|
|
|
|
|
|
|
|
|
@ -511,8 +499,7 @@ class Channel: |
|
|
|
request_serializer: Optional[SerializingFunction] = None, |
|
|
|
request_serializer: Optional[SerializingFunction] = None, |
|
|
|
response_deserializer: Optional[DeserializingFunction] = None |
|
|
|
response_deserializer: Optional[DeserializingFunction] = None |
|
|
|
) -> StreamUnaryMultiCallable: |
|
|
|
) -> StreamUnaryMultiCallable: |
|
|
|
return StreamUnaryMultiCallable(self._channel, self._ongoing_calls, |
|
|
|
return StreamUnaryMultiCallable(self._channel, _common.encode(method), |
|
|
|
_common.encode(method), |
|
|
|
|
|
|
|
request_serializer, |
|
|
|
request_serializer, |
|
|
|
response_deserializer, None, self._loop) |
|
|
|
response_deserializer, None, self._loop) |
|
|
|
|
|
|
|
|
|
|
@ -522,8 +509,7 @@ class Channel: |
|
|
|
request_serializer: Optional[SerializingFunction] = None, |
|
|
|
request_serializer: Optional[SerializingFunction] = None, |
|
|
|
response_deserializer: Optional[DeserializingFunction] = None |
|
|
|
response_deserializer: Optional[DeserializingFunction] = None |
|
|
|
) -> StreamStreamMultiCallable: |
|
|
|
) -> StreamStreamMultiCallable: |
|
|
|
return StreamStreamMultiCallable(self._channel, self._ongoing_calls, |
|
|
|
return StreamStreamMultiCallable(self._channel, _common.encode(method), |
|
|
|
_common.encode(method), |
|
|
|
|
|
|
|
request_serializer, |
|
|
|
request_serializer, |
|
|
|
response_deserializer, None, |
|
|
|
response_deserializer, None, |
|
|
|
self._loop) |
|
|
|
self._loop) |
|
|
|