forked from goodboy/tractor
1
0
Fork 0
tractor/tractor/_ipc.py

350 lines
9.9 KiB
Python
Raw Normal View History

2018-05-30 16:36:23 +00:00
"""
Inter-process comms abstractions
"""
from functools import partial
import struct
import typing
from typing import Any, Tuple, Optional
2018-05-30 16:36:23 +00:00
from tricycle import BufferedReceiveStream
2018-05-30 16:36:23 +00:00
import msgpack
import trio
2018-07-11 22:08:57 +00:00
from async_generator import asynccontextmanager
2018-05-30 16:36:23 +00:00
from .log import get_logger
from ._exceptions import TransportClosed
log = get_logger(__name__)
2018-05-30 16:36:23 +00:00
# :eyeroll:
try:
import msgpack_numpy
Unpacker = msgpack_numpy.Unpacker
except ImportError:
# just plain ``msgpack`` requires tweaking key settings
Unpacker = partial(msgpack.Unpacker, strict_map_key=False)
2018-05-30 16:36:23 +00:00
class MsgpackTCPStream:
'''A ``trio.SocketStream`` delivering ``msgpack`` formatted data
using ``msgpack-python``.
'''
def __init__(
self,
stream: trio.SocketStream,
) -> None:
2018-05-30 16:36:23 +00:00
self.stream = stream
2019-12-10 05:55:03 +00:00
assert self.stream.socket
2019-12-10 05:55:03 +00:00
# should both be IP sockets
lsockname = stream.socket.getsockname()
assert isinstance(lsockname, tuple)
self._laddr = lsockname[:2]
2019-12-10 05:55:03 +00:00
rsockname = stream.socket.getpeername()
assert isinstance(rsockname, tuple)
self._raddr = rsockname[:2]
2021-06-27 04:47:49 +00:00
# start first entry to read loop
2018-05-30 16:36:23 +00:00
self._agen = self._iter_packets()
2018-12-15 07:20:19 +00:00
self._send_lock = trio.StrictFIFOLock()
2018-05-30 16:36:23 +00:00
async def _iter_packets(self) -> typing.AsyncGenerator[dict, None]:
2018-05-30 16:36:23 +00:00
"""Yield packets from the underlying stream.
"""
unpacker = Unpacker(
raw=False,
use_list=False,
)
2018-05-30 16:36:23 +00:00
while True:
data = await self.stream.receive_some(2**10)
log.trace(f"received {data}") # type: ignore
2018-05-30 16:36:23 +00:00
if data == b'':
raise TransportClosed(
f'transport {self} was already closed prior ro read'
)
2018-05-30 16:36:23 +00:00
unpacker.feed(data)
for packet in unpacker:
yield packet
@property
2019-12-10 05:55:03 +00:00
def laddr(self) -> Tuple[Any, ...]:
return self._laddr
@property
2019-12-10 05:55:03 +00:00
def raddr(self) -> Tuple[Any, ...]:
return self._raddr
2019-12-10 05:55:03 +00:00
# XXX: should this instead be called `.sendall()`?
async def send(self, data: Any) -> None:
async with self._send_lock:
return await self.stream.send_all(
msgpack.dumps(data, use_bin_type=True)
)
2018-05-30 16:36:23 +00:00
async def recv(self) -> Any:
2018-05-30 16:36:23 +00:00
return await self._agen.asend(None)
2018-07-25 04:27:13 +00:00
def __aiter__(self):
2018-05-30 16:36:23 +00:00
return self._agen
def connected(self) -> bool:
2018-07-04 07:16:00 +00:00
return self.stream.socket.fileno() != -1
2018-05-30 16:36:23 +00:00
class MsgspecTCPStream(MsgpackTCPStream):
'''A ``trio.SocketStream`` delivering ``msgpack`` formatted data
using ``msgspec``.
'''
def __init__(
self,
stream: trio.SocketStream,
prefix_size: int = 4,
) -> None:
super().__init__(stream)
self.recv_stream = BufferedReceiveStream(transport_stream=stream)
self.prefix_size = prefix_size
import msgspec
# TODO: struct aware messaging coders
self.encode = msgspec.Encoder().encode
self.decode = msgspec.Decoder().decode # dict[str, Any])
async def _iter_packets(self) -> typing.AsyncGenerator[dict, None]:
"""Yield packets from the underlying stream.
"""
while True:
try:
header = await self.recv_stream.receive_exactly(4)
except (ValueError):
raise TransportClosed(
f'transport {self} was already closed prior ro read'
)
if header == b'':
raise TransportClosed(
f'transport {self} was already closed prior ro read'
)
size, = struct.unpack("<I", header)
2021-07-01 18:52:52 +00:00
log.trace(f'received header {size}') # type: ignore
msg_bytes = await self.recv_stream.receive_exactly(size)
log.trace(f"received {msg_bytes}") # type: ignore
yield self.decode(msg_bytes)
async def send(self, data: Any) -> None:
async with self._send_lock:
2021-07-01 18:52:52 +00:00
bytes_data: bytes = self.encode(data)
# supposedly the fastest says,
# https://stackoverflow.com/a/54027962
2021-07-01 18:52:52 +00:00
size: bytes = struct.pack("<I", len(bytes_data))
return await self.stream.send_all(size + bytes_data)
class Channel:
"""An inter-process channel for communication between (remote) actors.
2018-05-30 16:36:23 +00:00
Currently the only supported transport is a ``trio.SocketStream``.
2018-05-30 16:36:23 +00:00
"""
def __init__(
self,
2018-08-31 21:16:24 +00:00
destaddr: Optional[Tuple[str, int]] = None,
on_reconnect: typing.Callable[..., typing.Awaitable] = None,
auto_reconnect: bool = False,
stream: trio.SocketStream = None, # expected to be active
) -> None:
2018-05-30 16:36:23 +00:00
self._recon_seq = on_reconnect
self._autorecon = auto_reconnect
2021-07-01 18:52:52 +00:00
stream_serializer_type = MsgpackTCPStream
try:
# if installed load the msgspec transport since it's faster
import msgspec # noqa
2021-07-01 18:52:52 +00:00
stream_serializer_type = MsgspecTCPStream
except ImportError:
2021-07-01 18:52:52 +00:00
pass
self.stream_serializer_type = stream_serializer_type
2021-07-01 18:52:52 +00:00
self.msgstream = stream_serializer_type(stream) if stream else None
if self.msgstream and destaddr:
raise ValueError(
f"A stream was provided with local addr {self.laddr}"
)
self._destaddr = self.msgstream.raddr if self.msgstream else destaddr
2021-07-01 18:52:52 +00:00
2018-06-21 21:09:22 +00:00
# set after handshake - always uid of far end
2018-08-31 21:16:24 +00:00
self.uid: Optional[Tuple[str, str]] = None
2021-07-01 18:52:52 +00:00
2018-08-31 21:16:24 +00:00
# set if far end actor errors internally
self._exc: Optional[Exception] = None
self._agen = self._aiter_recv()
self._closed: bool = False
def __repr__(self) -> str:
if self.msgstream:
return repr(
2019-12-10 05:55:03 +00:00
self.msgstream.stream.socket._sock).replace( # type: ignore
"socket.socket", "Channel")
return object.__repr__(self)
@property
2019-12-10 05:55:03 +00:00
def laddr(self) -> Optional[Tuple[Any, ...]]:
return self.msgstream.laddr if self.msgstream else None
@property
2019-12-10 05:55:03 +00:00
def raddr(self) -> Optional[Tuple[Any, ...]]:
return self.msgstream.raddr if self.msgstream else None
async def connect(
self,
destaddr: Tuple[Any, ...] = None,
2019-12-10 05:55:03 +00:00
**kwargs
) -> trio.SocketStream:
2018-07-04 07:16:00 +00:00
if self.connected():
raise RuntimeError("channel is already connected?")
destaddr = destaddr or self._destaddr
2019-12-10 05:55:03 +00:00
assert isinstance(destaddr, tuple)
stream = await trio.open_tcp_stream(
*destaddr,
**kwargs
)
self.msgstream = self.stream_serializer_type(stream)
2018-05-30 16:36:23 +00:00
return stream
2018-08-31 21:16:24 +00:00
async def send(self, item: Any) -> None:
2018-08-31 21:16:24 +00:00
log.trace(f"send `{item}`") # type: ignore
assert self.msgstream
await self.msgstream.send(item)
2018-05-30 16:36:23 +00:00
2018-08-31 21:16:24 +00:00
async def recv(self) -> Any:
assert self.msgstream
2018-05-30 16:36:23 +00:00
try:
return await self.msgstream.recv()
2018-11-09 06:53:15 +00:00
except trio.BrokenResourceError:
2018-05-30 16:36:23 +00:00
if self._autorecon:
await self._reconnect()
return await self.recv()
raise
async def aclose(self) -> None:
2018-07-04 07:16:00 +00:00
log.debug(f"Closing {self}")
assert self.msgstream
await self.msgstream.stream.aclose()
self._closed = True
log.error(f'CLOSING CHAN {self}')
2018-05-30 16:36:23 +00:00
async def __aenter__(self):
await self.connect()
2018-05-30 16:36:23 +00:00
return self
async def __aexit__(self, *args):
await self.aclose(*args)
def __aiter__(self):
return self._agen
async def _reconnect(self) -> None:
2018-05-30 16:36:23 +00:00
"""Handle connection failures by polling until a reconnect can be
established.
"""
down = False
while True:
try:
with trio.move_on_after(3) as cancel_scope:
await self.connect()
cancelled = cancel_scope.cancelled_caught
if cancelled:
2018-09-10 19:19:49 +00:00
log.warning(
2018-05-30 16:36:23 +00:00
"Reconnect timed out after 3 seconds, retrying...")
continue
else:
2018-09-10 19:19:49 +00:00
log.warning("Stream connection re-established!")
2018-05-30 16:36:23 +00:00
# run any reconnection sequence
on_recon = self._recon_seq
if on_recon:
await on_recon(self)
2018-05-30 16:36:23 +00:00
break
except (OSError, ConnectionRefusedError):
if not down:
down = True
2018-09-10 19:19:49 +00:00
log.warning(
f"Connection to {self.raddr} went down, waiting"
2018-05-30 16:36:23 +00:00
" for re-establishment")
await trio.sleep(1)
async def _aiter_recv(
self
2018-08-31 21:16:24 +00:00
) -> typing.AsyncGenerator[Any, None]:
2018-05-30 16:36:23 +00:00
"""Async iterate items from underlying stream.
"""
assert self.msgstream
2018-05-30 16:36:23 +00:00
while True:
try:
async for item in self.msgstream:
2018-05-30 16:36:23 +00:00
yield item
# sent = yield item
# if sent is not None:
# # optimization, passing None through all the
# # time is pointless
# await self.msgstream.send(sent)
2018-11-09 06:53:15 +00:00
except trio.BrokenResourceError:
2018-05-30 16:36:23 +00:00
if not self._autorecon:
raise
2018-07-04 07:16:00 +00:00
await self.aclose()
2018-05-30 16:36:23 +00:00
if self._autorecon: # attempt reconnect
await self._reconnect()
continue
else:
return
2018-06-21 21:09:22 +00:00
def connected(self) -> bool:
return self.msgstream.connected() if self.msgstream else False
2018-07-11 22:08:57 +00:00
@asynccontextmanager
async def _connect_chan(
host: str, port: int
2018-08-31 21:16:24 +00:00
) -> typing.AsyncGenerator[Channel, None]:
"""Create and connect a channel with disconnect on context manager
teardown.
2018-07-11 22:08:57 +00:00
"""
chan = Channel((host, port))
await chan.connect()
yield chan
await chan.aclose()