mirror of https://gitee.com/openkylin/qemu.git
python/aqmp: add generic async message-based protocol support
This is the bare minimum that you need to establish a full-duplex async message-based protocol with Python's asyncio. The features to be added in forthcoming commits are: - Runstate tracking - Logging - Support for incoming connections via accept() - _cb_outbound, _cb_inbound message hooks - _readline() method Signed-off-by: John Snow <jsnow@redhat.com> Message-id: 20210915162955.333025-6-jsnow@redhat.com Signed-off-by: John Snow <jsnow@redhat.com>
This commit is contained in:
parent
a07616d612
commit
4ccaab0377
|
@ -22,10 +22,12 @@
|
||||||
# the COPYING file in the top-level directory.
|
# the COPYING file in the top-level directory.
|
||||||
|
|
||||||
from .error import AQMPError
|
from .error import AQMPError
|
||||||
|
from .protocol import ConnectError
|
||||||
|
|
||||||
|
|
||||||
# The order of these fields impact the Sphinx documentation order.
|
# The order of these fields impact the Sphinx documentation order.
|
||||||
__all__ = (
|
__all__ = (
|
||||||
# Exceptions
|
# Exceptions, most generic to most explicit
|
||||||
'AQMPError',
|
'AQMPError',
|
||||||
|
'ConnectError',
|
||||||
)
|
)
|
||||||
|
|
|
@ -0,0 +1,521 @@
|
||||||
|
"""
|
||||||
|
Generic Asynchronous Message-based Protocol Support
|
||||||
|
|
||||||
|
This module provides a generic framework for sending and receiving
|
||||||
|
messages over an asyncio stream. `AsyncProtocol` is an abstract class
|
||||||
|
that implements the core mechanisms of a simple send/receive protocol,
|
||||||
|
and is designed to be extended.
|
||||||
|
|
||||||
|
In this package, it is used as the implementation for the `QMPClient`
|
||||||
|
class.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
from asyncio import StreamReader, StreamWriter
|
||||||
|
from ssl import SSLContext
|
||||||
|
# import exceptions will be removed in a forthcoming commit.
|
||||||
|
# The problem stems from pylint/flake8 believing that 'Any'
|
||||||
|
# is unused because of its only use in a string-quoted type.
|
||||||
|
from typing import ( # pylint: disable=unused-import # noqa
|
||||||
|
Any,
|
||||||
|
Awaitable,
|
||||||
|
Callable,
|
||||||
|
Generic,
|
||||||
|
List,
|
||||||
|
Optional,
|
||||||
|
Tuple,
|
||||||
|
TypeVar,
|
||||||
|
Union,
|
||||||
|
)
|
||||||
|
|
||||||
|
from .error import AQMPError
|
||||||
|
from .util import (
|
||||||
|
bottom_half,
|
||||||
|
create_task,
|
||||||
|
flush,
|
||||||
|
is_closing,
|
||||||
|
upper_half,
|
||||||
|
wait_closed,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
T = TypeVar('T')
|
||||||
|
_TaskFN = Callable[[], Awaitable[None]] # aka ``async def func() -> None``
|
||||||
|
_FutureT = TypeVar('_FutureT', bound=Optional['asyncio.Future[Any]'])
|
||||||
|
|
||||||
|
|
||||||
|
class ConnectError(AQMPError):
|
||||||
|
"""
|
||||||
|
Raised when the initial connection process has failed.
|
||||||
|
|
||||||
|
This Exception always wraps a "root cause" exception that can be
|
||||||
|
interrogated for additional information.
|
||||||
|
|
||||||
|
:param error_message: Human-readable string describing the error.
|
||||||
|
:param exc: The root-cause exception.
|
||||||
|
"""
|
||||||
|
def __init__(self, error_message: str, exc: Exception):
|
||||||
|
super().__init__(error_message)
|
||||||
|
#: Human-readable error string
|
||||||
|
self.error_message: str = error_message
|
||||||
|
#: Wrapped root cause exception
|
||||||
|
self.exc: Exception = exc
|
||||||
|
|
||||||
|
def __str__(self) -> str:
|
||||||
|
return f"{self.error_message}: {self.exc!s}"
|
||||||
|
|
||||||
|
|
||||||
|
class AsyncProtocol(Generic[T]):
|
||||||
|
"""
|
||||||
|
AsyncProtocol implements a generic async message-based protocol.
|
||||||
|
|
||||||
|
This protocol assumes the basic unit of information transfer between
|
||||||
|
client and server is a "message", the details of which are left up
|
||||||
|
to the implementation. It assumes the sending and receiving of these
|
||||||
|
messages is full-duplex and not necessarily correlated; i.e. it
|
||||||
|
supports asynchronous inbound messages.
|
||||||
|
|
||||||
|
It is designed to be extended by a specific protocol which provides
|
||||||
|
the implementations for how to read and send messages. These must be
|
||||||
|
defined in `_do_recv()` and `_do_send()`, respectively.
|
||||||
|
|
||||||
|
Other callbacks have a default implementation, but are intended to be
|
||||||
|
either extended or overridden:
|
||||||
|
|
||||||
|
- `_establish_session`:
|
||||||
|
The base implementation starts the reader/writer tasks.
|
||||||
|
A protocol implementation can override this call, inserting
|
||||||
|
actions to be taken prior to starting the reader/writer tasks
|
||||||
|
before the super() call; actions needing to occur afterwards
|
||||||
|
can be written after the super() call.
|
||||||
|
- `_on_message`:
|
||||||
|
Actions to be performed when a message is received.
|
||||||
|
"""
|
||||||
|
# pylint: disable=too-many-instance-attributes
|
||||||
|
|
||||||
|
# -------------------------
|
||||||
|
# Section: Public interface
|
||||||
|
# -------------------------
|
||||||
|
|
||||||
|
def __init__(self) -> None:
|
||||||
|
# stream I/O
|
||||||
|
self._reader: Optional[StreamReader] = None
|
||||||
|
self._writer: Optional[StreamWriter] = None
|
||||||
|
|
||||||
|
# Outbound Message queue
|
||||||
|
self._outgoing: asyncio.Queue[T]
|
||||||
|
|
||||||
|
# Special, long-running tasks:
|
||||||
|
self._reader_task: Optional[asyncio.Future[None]] = None
|
||||||
|
self._writer_task: Optional[asyncio.Future[None]] = None
|
||||||
|
|
||||||
|
# Aggregate of the above two tasks, used for Exception management.
|
||||||
|
self._bh_tasks: Optional[asyncio.Future[Tuple[None, None]]] = None
|
||||||
|
|
||||||
|
#: Disconnect task. The disconnect implementation runs in a task
|
||||||
|
#: so that asynchronous disconnects (initiated by the
|
||||||
|
#: reader/writer) are allowed to wait for the reader/writers to
|
||||||
|
#: exit.
|
||||||
|
self._dc_task: Optional[asyncio.Future[None]] = None
|
||||||
|
|
||||||
|
@upper_half
|
||||||
|
async def connect(self, address: Union[str, Tuple[str, int]],
|
||||||
|
ssl: Optional[SSLContext] = None) -> None:
|
||||||
|
"""
|
||||||
|
Connect to the server and begin processing message queues.
|
||||||
|
|
||||||
|
If this call fails, `runstate` is guaranteed to be set back to `IDLE`.
|
||||||
|
|
||||||
|
:param address:
|
||||||
|
Address to connect to; UNIX socket path or TCP address/port.
|
||||||
|
:param ssl: SSL context to use, if any.
|
||||||
|
|
||||||
|
:raise StateError: When the `Runstate` is not `IDLE`.
|
||||||
|
:raise ConnectError: If a connection cannot be made to the server.
|
||||||
|
"""
|
||||||
|
await self._new_session(address, ssl)
|
||||||
|
|
||||||
|
@upper_half
|
||||||
|
async def disconnect(self) -> None:
|
||||||
|
"""
|
||||||
|
Disconnect and wait for all tasks to fully stop.
|
||||||
|
|
||||||
|
If there was an exception that caused the reader/writers to
|
||||||
|
terminate prematurely, it will be raised here.
|
||||||
|
|
||||||
|
:raise Exception: When the reader or writer terminate unexpectedly.
|
||||||
|
"""
|
||||||
|
self._schedule_disconnect()
|
||||||
|
await self._wait_disconnect()
|
||||||
|
|
||||||
|
# --------------------------
|
||||||
|
# Section: Session machinery
|
||||||
|
# --------------------------
|
||||||
|
|
||||||
|
@upper_half
|
||||||
|
async def _new_session(self,
|
||||||
|
address: Union[str, Tuple[str, int]],
|
||||||
|
ssl: Optional[SSLContext] = None) -> None:
|
||||||
|
"""
|
||||||
|
Establish a new connection and initialize the session.
|
||||||
|
|
||||||
|
Connect or accept a new connection, then begin the protocol
|
||||||
|
session machinery. If this call fails, `runstate` is guaranteed
|
||||||
|
to be set back to `IDLE`.
|
||||||
|
|
||||||
|
:param address:
|
||||||
|
Address to connect to;
|
||||||
|
UNIX socket path or TCP address/port.
|
||||||
|
:param ssl: SSL context to use, if any.
|
||||||
|
|
||||||
|
:raise ConnectError:
|
||||||
|
When a connection or session cannot be established.
|
||||||
|
|
||||||
|
This exception will wrap a more concrete one. In most cases,
|
||||||
|
the wrapped exception will be `OSError` or `EOFError`. If a
|
||||||
|
protocol-level failure occurs while establishing a new
|
||||||
|
session, the wrapped error may also be an `AQMPError`.
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
phase = "connection"
|
||||||
|
await self._establish_connection(address, ssl)
|
||||||
|
|
||||||
|
phase = "session"
|
||||||
|
await self._establish_session()
|
||||||
|
|
||||||
|
except BaseException as err:
|
||||||
|
emsg = f"Failed to establish {phase}"
|
||||||
|
await self.disconnect()
|
||||||
|
|
||||||
|
# NB: CancelledError is not a BaseException before Python 3.8
|
||||||
|
if isinstance(err, asyncio.CancelledError):
|
||||||
|
raise
|
||||||
|
|
||||||
|
if isinstance(err, Exception):
|
||||||
|
raise ConnectError(emsg, err) from err
|
||||||
|
|
||||||
|
# Raise BaseExceptions un-wrapped, they're more important.
|
||||||
|
raise
|
||||||
|
|
||||||
|
@upper_half
|
||||||
|
async def _establish_connection(
|
||||||
|
self,
|
||||||
|
address: Union[str, Tuple[str, int]],
|
||||||
|
ssl: Optional[SSLContext] = None,
|
||||||
|
) -> None:
|
||||||
|
"""
|
||||||
|
Establish a new connection.
|
||||||
|
|
||||||
|
:param address:
|
||||||
|
Address to connect to/listen on;
|
||||||
|
UNIX socket path or TCP address/port.
|
||||||
|
:param ssl: SSL context to use, if any.
|
||||||
|
"""
|
||||||
|
await self._do_connect(address, ssl)
|
||||||
|
|
||||||
|
@upper_half
|
||||||
|
async def _do_connect(self, address: Union[str, Tuple[str, int]],
|
||||||
|
ssl: Optional[SSLContext] = None) -> None:
|
||||||
|
"""
|
||||||
|
Acting as the transport client, initiate a connection to a server.
|
||||||
|
|
||||||
|
:param address:
|
||||||
|
Address to connect to; UNIX socket path or TCP address/port.
|
||||||
|
:param ssl: SSL context to use, if any.
|
||||||
|
|
||||||
|
:raise OSError: For stream-related errors.
|
||||||
|
"""
|
||||||
|
if isinstance(address, tuple):
|
||||||
|
connect = asyncio.open_connection(address[0], address[1], ssl=ssl)
|
||||||
|
else:
|
||||||
|
connect = asyncio.open_unix_connection(path=address, ssl=ssl)
|
||||||
|
self._reader, self._writer = await connect
|
||||||
|
|
||||||
|
@upper_half
|
||||||
|
async def _establish_session(self) -> None:
|
||||||
|
"""
|
||||||
|
Establish a new session.
|
||||||
|
|
||||||
|
Starts the readers/writer tasks; subclasses may perform their
|
||||||
|
own negotiations here. The Runstate will be RUNNING upon
|
||||||
|
successful conclusion.
|
||||||
|
"""
|
||||||
|
self._outgoing = asyncio.Queue()
|
||||||
|
|
||||||
|
reader_coro = self._bh_loop_forever(self._bh_recv_message)
|
||||||
|
writer_coro = self._bh_loop_forever(self._bh_send_message)
|
||||||
|
|
||||||
|
self._reader_task = create_task(reader_coro)
|
||||||
|
self._writer_task = create_task(writer_coro)
|
||||||
|
|
||||||
|
self._bh_tasks = asyncio.gather(
|
||||||
|
self._reader_task,
|
||||||
|
self._writer_task,
|
||||||
|
)
|
||||||
|
|
||||||
|
@upper_half
|
||||||
|
@bottom_half
|
||||||
|
def _schedule_disconnect(self) -> None:
|
||||||
|
"""
|
||||||
|
Initiate a disconnect; idempotent.
|
||||||
|
|
||||||
|
This method is used both in the upper-half as a direct
|
||||||
|
consequence of `disconnect()`, and in the bottom-half in the
|
||||||
|
case of unhandled exceptions in the reader/writer tasks.
|
||||||
|
|
||||||
|
It can be invoked no matter what the `runstate` is.
|
||||||
|
"""
|
||||||
|
if not self._dc_task:
|
||||||
|
self._dc_task = create_task(self._bh_disconnect())
|
||||||
|
|
||||||
|
@upper_half
|
||||||
|
async def _wait_disconnect(self) -> None:
|
||||||
|
"""
|
||||||
|
Waits for a previously scheduled disconnect to finish.
|
||||||
|
|
||||||
|
This method will gather any bottom half exceptions and re-raise
|
||||||
|
the one that occurred first; presuming it to be the root cause
|
||||||
|
of any subsequent Exceptions. It is intended to be used in the
|
||||||
|
upper half of the call chain.
|
||||||
|
|
||||||
|
:raise Exception:
|
||||||
|
Arbitrary exception re-raised on behalf of the reader/writer.
|
||||||
|
"""
|
||||||
|
assert self._dc_task
|
||||||
|
|
||||||
|
aws: List[Awaitable[object]] = [self._dc_task]
|
||||||
|
if self._bh_tasks:
|
||||||
|
aws.insert(0, self._bh_tasks)
|
||||||
|
all_defined_tasks = asyncio.gather(*aws)
|
||||||
|
|
||||||
|
# Ensure disconnect is done; Exception (if any) is not raised here:
|
||||||
|
await asyncio.wait((self._dc_task,))
|
||||||
|
|
||||||
|
try:
|
||||||
|
await all_defined_tasks # Raise Exceptions from the bottom half.
|
||||||
|
finally:
|
||||||
|
self._cleanup()
|
||||||
|
|
||||||
|
@upper_half
|
||||||
|
def _cleanup(self) -> None:
|
||||||
|
"""
|
||||||
|
Fully reset this object to a clean state and return to `IDLE`.
|
||||||
|
"""
|
||||||
|
def _paranoid_task_erase(task: _FutureT) -> Optional[_FutureT]:
|
||||||
|
# Help to erase a task, ENSURING it is fully quiesced first.
|
||||||
|
assert (task is None) or task.done()
|
||||||
|
return None if (task and task.done()) else task
|
||||||
|
|
||||||
|
self._dc_task = _paranoid_task_erase(self._dc_task)
|
||||||
|
self._reader_task = _paranoid_task_erase(self._reader_task)
|
||||||
|
self._writer_task = _paranoid_task_erase(self._writer_task)
|
||||||
|
self._bh_tasks = _paranoid_task_erase(self._bh_tasks)
|
||||||
|
|
||||||
|
self._reader = None
|
||||||
|
self._writer = None
|
||||||
|
|
||||||
|
# ----------------------------
|
||||||
|
# Section: Bottom Half methods
|
||||||
|
# ----------------------------
|
||||||
|
|
||||||
|
@bottom_half
|
||||||
|
async def _bh_disconnect(self) -> None:
|
||||||
|
"""
|
||||||
|
Disconnect and cancel all outstanding tasks.
|
||||||
|
|
||||||
|
It is designed to be called from its task context,
|
||||||
|
:py:obj:`~AsyncProtocol._dc_task`. By running in its own task,
|
||||||
|
it is free to wait on any pending actions that may still need to
|
||||||
|
occur in either the reader or writer tasks.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def _done(task: Optional['asyncio.Future[Any]']) -> bool:
|
||||||
|
return task is not None and task.done()
|
||||||
|
|
||||||
|
# NB: We can't rely on _bh_tasks being done() here, it may not
|
||||||
|
# yet have had a chance to run and gather itself.
|
||||||
|
tasks = tuple(filter(None, (self._writer_task, self._reader_task)))
|
||||||
|
error_pathway = _done(self._reader_task) or _done(self._writer_task)
|
||||||
|
|
||||||
|
try:
|
||||||
|
# Try to flush the writer, if possible:
|
||||||
|
if not error_pathway:
|
||||||
|
await self._bh_flush_writer()
|
||||||
|
except:
|
||||||
|
error_pathway = True
|
||||||
|
raise
|
||||||
|
finally:
|
||||||
|
# Cancel any still-running tasks:
|
||||||
|
if self._writer_task is not None and not self._writer_task.done():
|
||||||
|
self._writer_task.cancel()
|
||||||
|
if self._reader_task is not None and not self._reader_task.done():
|
||||||
|
self._reader_task.cancel()
|
||||||
|
|
||||||
|
# Close out the tasks entirely (Won't raise):
|
||||||
|
if tasks:
|
||||||
|
await asyncio.wait(tasks)
|
||||||
|
|
||||||
|
# Lastly, close the stream itself. (May raise):
|
||||||
|
await self._bh_close_stream(error_pathway)
|
||||||
|
|
||||||
|
@bottom_half
|
||||||
|
async def _bh_flush_writer(self) -> None:
|
||||||
|
if not self._writer_task:
|
||||||
|
return
|
||||||
|
|
||||||
|
await self._outgoing.join()
|
||||||
|
if self._writer is not None:
|
||||||
|
await flush(self._writer)
|
||||||
|
|
||||||
|
@bottom_half
|
||||||
|
async def _bh_close_stream(self, error_pathway: bool = False) -> None:
|
||||||
|
# NB: Closing the writer also implcitly closes the reader.
|
||||||
|
if not self._writer:
|
||||||
|
return
|
||||||
|
|
||||||
|
if not is_closing(self._writer):
|
||||||
|
self._writer.close()
|
||||||
|
|
||||||
|
try:
|
||||||
|
await wait_closed(self._writer)
|
||||||
|
except Exception: # pylint: disable=broad-except
|
||||||
|
# It's hard to tell if the Stream is already closed or
|
||||||
|
# not. Even if one of the tasks has failed, it may have
|
||||||
|
# failed for a higher-layered protocol reason. The
|
||||||
|
# stream could still be open and perfectly fine.
|
||||||
|
# I don't know how to discern its health here.
|
||||||
|
|
||||||
|
if error_pathway:
|
||||||
|
# We already know that *something* went wrong. Let's
|
||||||
|
# just trust that the Exception we already have is the
|
||||||
|
# better one to present to the user, even if we don't
|
||||||
|
# genuinely *know* the relationship between the two.
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
# Oops, this is a brand-new error!
|
||||||
|
raise
|
||||||
|
|
||||||
|
@bottom_half
|
||||||
|
async def _bh_loop_forever(self, async_fn: _TaskFN) -> None:
|
||||||
|
"""
|
||||||
|
Run one of the bottom-half methods in a loop forever.
|
||||||
|
|
||||||
|
If the bottom half ever raises any exception, schedule a
|
||||||
|
disconnect that will terminate the entire loop.
|
||||||
|
|
||||||
|
:param async_fn: The bottom-half method to run in a loop.
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
while True:
|
||||||
|
await async_fn()
|
||||||
|
except asyncio.CancelledError:
|
||||||
|
# We have been cancelled by _bh_disconnect, exit gracefully.
|
||||||
|
return
|
||||||
|
except BaseException:
|
||||||
|
self._schedule_disconnect()
|
||||||
|
raise
|
||||||
|
|
||||||
|
@bottom_half
|
||||||
|
async def _bh_send_message(self) -> None:
|
||||||
|
"""
|
||||||
|
Wait for an outgoing message, then send it.
|
||||||
|
|
||||||
|
Designed to be run in `_bh_loop_forever()`.
|
||||||
|
"""
|
||||||
|
msg = await self._outgoing.get()
|
||||||
|
try:
|
||||||
|
await self._send(msg)
|
||||||
|
finally:
|
||||||
|
self._outgoing.task_done()
|
||||||
|
|
||||||
|
@bottom_half
|
||||||
|
async def _bh_recv_message(self) -> None:
|
||||||
|
"""
|
||||||
|
Wait for an incoming message and call `_on_message` to route it.
|
||||||
|
|
||||||
|
Designed to be run in `_bh_loop_forever()`.
|
||||||
|
"""
|
||||||
|
msg = await self._recv()
|
||||||
|
await self._on_message(msg)
|
||||||
|
|
||||||
|
# --------------------
|
||||||
|
# Section: Message I/O
|
||||||
|
# --------------------
|
||||||
|
|
||||||
|
@upper_half
|
||||||
|
@bottom_half
|
||||||
|
async def _do_recv(self) -> T:
|
||||||
|
"""
|
||||||
|
Abstract: Read from the stream and return a message.
|
||||||
|
|
||||||
|
Very low-level; intended to only be called by `_recv()`.
|
||||||
|
"""
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
@upper_half
|
||||||
|
@bottom_half
|
||||||
|
async def _recv(self) -> T:
|
||||||
|
"""
|
||||||
|
Read an arbitrary protocol message.
|
||||||
|
|
||||||
|
.. warning::
|
||||||
|
This method is intended primarily for `_bh_recv_message()`
|
||||||
|
to use in an asynchronous task loop. Using it outside of
|
||||||
|
this loop will "steal" messages from the normal routing
|
||||||
|
mechanism. It is safe to use prior to `_establish_session()`,
|
||||||
|
but should not be used otherwise.
|
||||||
|
|
||||||
|
This method uses `_do_recv()` to retrieve the raw message, and
|
||||||
|
then transforms it using `_cb_inbound()`.
|
||||||
|
|
||||||
|
:return: A single (filtered, processed) protocol message.
|
||||||
|
"""
|
||||||
|
# A forthcoming commit makes this method less trivial.
|
||||||
|
return await self._do_recv()
|
||||||
|
|
||||||
|
@upper_half
|
||||||
|
@bottom_half
|
||||||
|
def _do_send(self, msg: T) -> None:
|
||||||
|
"""
|
||||||
|
Abstract: Write a message to the stream.
|
||||||
|
|
||||||
|
Very low-level; intended to only be called by `_send()`.
|
||||||
|
"""
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
@upper_half
|
||||||
|
@bottom_half
|
||||||
|
async def _send(self, msg: T) -> None:
|
||||||
|
"""
|
||||||
|
Send an arbitrary protocol message.
|
||||||
|
|
||||||
|
This method will transform any outgoing messages according to
|
||||||
|
`_cb_outbound()`.
|
||||||
|
|
||||||
|
.. warning::
|
||||||
|
Like `_recv()`, this method is intended to be called by
|
||||||
|
the writer task loop that processes outgoing
|
||||||
|
messages. Calling it directly may circumvent logic
|
||||||
|
implemented by the caller meant to correlate outgoing and
|
||||||
|
incoming messages.
|
||||||
|
|
||||||
|
:raise OSError: For problems with the underlying stream.
|
||||||
|
"""
|
||||||
|
# A forthcoming commit makes this method less trivial.
|
||||||
|
self._do_send(msg)
|
||||||
|
|
||||||
|
@bottom_half
|
||||||
|
async def _on_message(self, msg: T) -> None:
|
||||||
|
"""
|
||||||
|
Called to handle the receipt of a new message.
|
||||||
|
|
||||||
|
.. caution::
|
||||||
|
This is executed from within the reader loop, so be advised
|
||||||
|
that waiting on either the reader or writer task will lead
|
||||||
|
to deadlock. Additionally, any unhandled exceptions will
|
||||||
|
directly cause the loop to halt, so logic may be best-kept
|
||||||
|
to a minimum if at all possible.
|
||||||
|
|
||||||
|
:param msg: The incoming message
|
||||||
|
"""
|
||||||
|
# Nothing to do in the abstract case.
|
|
@ -13,12 +13,65 @@
|
||||||
Coroutine,
|
Coroutine,
|
||||||
Optional,
|
Optional,
|
||||||
TypeVar,
|
TypeVar,
|
||||||
|
cast,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
T = TypeVar('T')
|
T = TypeVar('T')
|
||||||
|
|
||||||
|
|
||||||
|
# --------------------------
|
||||||
|
# Section: Utility Functions
|
||||||
|
# --------------------------
|
||||||
|
|
||||||
|
|
||||||
|
async def flush(writer: asyncio.StreamWriter) -> None:
|
||||||
|
"""
|
||||||
|
Utility function to ensure a StreamWriter is *fully* drained.
|
||||||
|
|
||||||
|
`asyncio.StreamWriter.drain` only promises we will return to below
|
||||||
|
the "high-water mark". This function ensures we flush the entire
|
||||||
|
buffer -- by setting the high water mark to 0 and then calling
|
||||||
|
drain. The flow control limits are restored after the call is
|
||||||
|
completed.
|
||||||
|
"""
|
||||||
|
transport = cast(asyncio.WriteTransport, writer.transport)
|
||||||
|
|
||||||
|
# https://github.com/python/typeshed/issues/5779
|
||||||
|
low, high = transport.get_write_buffer_limits() # type: ignore
|
||||||
|
transport.set_write_buffer_limits(0, 0)
|
||||||
|
try:
|
||||||
|
await writer.drain()
|
||||||
|
finally:
|
||||||
|
transport.set_write_buffer_limits(high, low)
|
||||||
|
|
||||||
|
|
||||||
|
def upper_half(func: T) -> T:
|
||||||
|
"""
|
||||||
|
Do-nothing decorator that annotates a method as an "upper-half" method.
|
||||||
|
|
||||||
|
These methods must not call bottom-half functions directly, but can
|
||||||
|
schedule them to run.
|
||||||
|
"""
|
||||||
|
return func
|
||||||
|
|
||||||
|
|
||||||
|
def bottom_half(func: T) -> T:
|
||||||
|
"""
|
||||||
|
Do-nothing decorator that annotates a method as a "bottom-half" method.
|
||||||
|
|
||||||
|
These methods must take great care to handle their own exceptions whenever
|
||||||
|
possible. If they go unhandled, they will cause termination of the loop.
|
||||||
|
|
||||||
|
These methods do not, in general, have the ability to directly
|
||||||
|
report information to a caller’s context and will usually be
|
||||||
|
collected as a Task result instead.
|
||||||
|
|
||||||
|
They must not call upper-half functions directly.
|
||||||
|
"""
|
||||||
|
return func
|
||||||
|
|
||||||
|
|
||||||
# -------------------------------
|
# -------------------------------
|
||||||
# Section: Compatibility Wrappers
|
# Section: Compatibility Wrappers
|
||||||
# -------------------------------
|
# -------------------------------
|
||||||
|
|
Loading…
Reference in New Issue