-
Notifications
You must be signed in to change notification settings - Fork 577
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Support async predictors (✨ again ✨) (#2025)
* Revert "Revert "Support async predictors (#2010)" (#2022)" This reverts commit 8333a83. * Use anync stream redirector in setup so that the sync stream redirector context is only entered once, as this is a known source of problems associated with stdout/stderr orphaning. * Do not assert that writes from C are captured during setup * Do not wrap empty data in Log events * Exclude invalid output paths from infrastructure errors (#2030) * Exclude invalid output paths from infrastructure errors Closes PLAT-380 * Fix words given pluralization Co-authored-by: F <[email protected]> Signed-off-by: Dan Buch <[email protected]> --------- Signed-off-by: Dan Buch <[email protected]> Co-authored-by: F <[email protected]> --------- Signed-off-by: Dan Buch <[email protected]> Co-authored-by: F <[email protected]>
- Loading branch information
1 parent
f8e3461
commit 5eb31ff
Showing
9 changed files
with
415 additions
and
49 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,91 @@ | ||
import asyncio | ||
import multiprocessing | ||
from multiprocessing.connection import Connection | ||
from typing import Any, Optional | ||
|
||
from typing_extensions import Buffer | ||
|
||
_spawn = multiprocessing.get_context("spawn") | ||
|
||
|
||
class AsyncConnection: | ||
def __init__(self, connection: Connection) -> None: | ||
self._connection = connection | ||
self._event = asyncio.Event() | ||
loop = asyncio.get_event_loop() | ||
loop.add_reader(self._connection.fileno(), self._event.set) | ||
|
||
def send(self, obj: Any) -> None: | ||
"""Send a (picklable) object""" | ||
|
||
self._connection.send(obj) | ||
|
||
async def _wait_for_input(self) -> None: | ||
"""Wait until there is an input available to be read""" | ||
|
||
while not self._connection.poll(): | ||
await self._event.wait() | ||
self._event.clear() | ||
|
||
async def recv(self) -> Any: | ||
"""Receive a (picklable) object""" | ||
|
||
await self._wait_for_input() | ||
return self._connection.recv() | ||
|
||
def fileno(self) -> int: | ||
"""File descriptor or handle of the connection""" | ||
return self._connection.fileno() | ||
|
||
def close(self) -> None: | ||
"""Close the connection""" | ||
self._connection.close() | ||
|
||
async def poll(self, timeout: float = 0.0) -> bool: | ||
"""Whether there is an input available to be read""" | ||
|
||
if self._connection.poll(): | ||
return True | ||
|
||
try: | ||
await asyncio.wait_for(self._wait_for_input(), timeout=timeout) | ||
except asyncio.TimeoutError: | ||
return False | ||
return self._connection.poll() | ||
|
||
def send_bytes( | ||
self, buf: Buffer, offset: int = 0, size: Optional[int] = None | ||
) -> None: | ||
"""Send the bytes data from a bytes-like object""" | ||
|
||
self._connection.send_bytes(buf, offset, size) | ||
|
||
async def recv_bytes(self, maxlength: Optional[int] = None) -> bytes: | ||
""" | ||
Receive bytes data as a bytes object. | ||
""" | ||
|
||
await self._wait_for_input() | ||
return self._connection.recv_bytes(maxlength) | ||
|
||
async def recv_bytes_into(self, buf: Buffer, offset: int = 0) -> int: | ||
""" | ||
Receive bytes data into a writeable bytes-like object. | ||
Return the number of bytes read. | ||
""" | ||
|
||
await self._wait_for_input() | ||
return self._connection.recv_bytes_into(buf, offset) | ||
|
||
|
||
class LockedConnection: | ||
def __init__(self, connection: Connection) -> None: | ||
self.connection = connection | ||
self._lock = _spawn.Lock() | ||
|
||
def send(self, obj: Any) -> None: | ||
with self._lock: | ||
self.connection.send(obj) | ||
|
||
def recv(self) -> Any: | ||
return self.connection.recv() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.