From ce201d86fc3e0b28a1a3252d33001e474abad4ba Mon Sep 17 00:00:00 2001 From: Carl Montanari Date: Sat, 4 May 2024 13:07:13 -0700 Subject: [PATCH] refactor: override core scrapli reads in async/sync channels --- scrapli_netconf/channel/async_channel.py | 25 ++++++++++++++++++++---- scrapli_netconf/channel/sync_channel.py | 23 +++++++++++++++++++--- 2 files changed, 41 insertions(+), 7 deletions(-) diff --git a/scrapli_netconf/channel/async_channel.py b/scrapli_netconf/channel/async_channel.py index 6f7b943..c94ef9d 100644 --- a/scrapli_netconf/channel/async_channel.py +++ b/scrapli_netconf/channel/async_channel.py @@ -98,10 +98,17 @@ async def _send_client_capabilities( async def read(self) -> bytes: """ - Read chunks of output from the channel + Read chunks of output from the channels - Prior to super-ing "normal" scrapli read, check if there is anything on our read_buf, if - there is, return that first + Prior to doing "normal" scrapli read things (*), check if there is anything on our read_buf, + if there is, return that first. Historically this actually did do "normal" scrapli read + things by super-ing read, but, "normal" scrapli replaces \r it finds in output which can + cause nc1.1 chunk parsing to fail since we've removed some chars that were counted as part + of the chunk. so, now we just copypasta the normal read things here without that .replace, + and no longer super. + + Note that to *not* have the \r in output from system transport you will need >=2024.07.30 + scrapli version, but replacing the super-ing with this sorts out other transports! Args: N/A @@ -118,7 +125,17 @@ async def read(self) -> bytes: self._read_buf = b"" return read_buf - return await super().read() + buf = await self.transport.read() + + self.logger.debug(f"read: {buf!r}") + + if self.channel_log: + self.channel_log.write(buf) + + if b"\x1b" in buf.lower(): + buf = self._strip_ansi(buf=buf) + + return buf async def _read_until_input(self, channel_input: bytes) -> bytes: """ diff --git a/scrapli_netconf/channel/sync_channel.py b/scrapli_netconf/channel/sync_channel.py index 260e20b..3cb044c 100644 --- a/scrapli_netconf/channel/sync_channel.py +++ b/scrapli_netconf/channel/sync_channel.py @@ -200,8 +200,15 @@ def read(self) -> bytes: """ Read chunks of output from the channel - Prior to super-ing "normal" scrapli read, check if there is anything on our read_buf, if - there is, return that first + Prior to doing "normal" scrapli read things (*), check if there is anything on our read_buf, + if there is, return that first. Historically this actually did do "normal" scrapli read + things by super-ing read, but, "normal" scrapli replaces \r it finds in output which can + cause nc1.1 chunk parsing to fail since we've removed some chars that were counted as part + of the chunk. so, now we just copypasta the normal read things here without that .replace, + and no longer super. + + Note that to *not* have the \r in output from system transport you will need >=2024.07.30 + scrapli version, but replacing the super-ing with this sorts out other transports! Args: N/A @@ -218,7 +225,17 @@ def read(self) -> bytes: self._read_buf = b"" return read_buf - return super().read() + buf = self.transport.read() + + self.logger.debug(f"read: {buf!r}") + + if self.channel_log: + self.channel_log.write(buf) + + if b"\x1b" in buf.lower(): + buf = self._strip_ansi(buf=buf) + + return buf def _read_until_input(self, channel_input: bytes) -> bytes: """