# ========================================================================== # # # # KVMD - The main PiKVM daemon. # # # # Copyright (C) 2020 Maxim Devaev # # # # This program is free software: you can redistribute it and/or modify # # it under the terms of the GNU General Public License as published by # # the Free Software Foundation, either version 3 of the License, or # # (at your option) any later version. # # # # This program is distributed in the hope that it will be useful, # # but WITHOUT ANY WARRANTY; without even the implied warranty of # # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # # GNU General Public License for more details. # # # # You should have received a copy of the GNU General Public License # # along with this program. If not, see . # # # # ========================================================================== # import asyncio import ssl import struct from .... import aiotools from .errors import RfbConnectionError # ===== def rfb_format_remote(writer: asyncio.StreamWriter) -> str: return "[%s]:%d" % (writer.transport.get_extra_info("peername")[:2]) class RfbClientStream: def __init__(self, reader: asyncio.StreamReader, writer: asyncio.StreamWriter) -> None: self.__reader = reader self.__writer = writer self._remote = rfb_format_remote(writer) # ===== async def _read_number(self, msg: str, fmt: str) -> int: assert len(fmt) == 1 try: if fmt == "B": return (await self.__reader.readexactly(1))[0] else: fmt = f">{fmt}" return struct.unpack(fmt, await self.__reader.readexactly(struct.calcsize(fmt)))[0] except (ConnectionError, asyncio.IncompleteReadError) as err: raise RfbConnectionError(f"Can't read {msg}", err) async def _read_struct(self, msg: str, fmt: str) -> tuple[int, ...]: assert len(fmt) > 1 try: fmt = f">{fmt}" return struct.unpack(fmt, (await self.__reader.readexactly(struct.calcsize(fmt)))) except (ConnectionError, asyncio.IncompleteReadError) as err: raise RfbConnectionError(f"Can't read {msg}", err) async def _read_text(self, msg: str, length: int) -> str: try: return (await self.__reader.readexactly(length)).decode("utf-8", errors="ignore") except (ConnectionError, asyncio.IncompleteReadError) as err: raise RfbConnectionError(f"Can't read {msg}", err) # ===== async def _write_struct(self, msg: str, fmt: str, *values: (int | bytes), drain: bool=True) -> None: try: if not fmt: for value in values: assert isinstance(value, bytes) self.__writer.write(value) elif fmt == "B": assert len(values) == 1 assert isinstance(values[0], int) self.__writer.write(bytes([values[0]])) else: self.__writer.write(struct.pack(f">{fmt}", *values)) if drain: await self.__writer.drain() except ConnectionError as err: raise RfbConnectionError(f"Can't write {msg}", err) async def _write_reason(self, msg: str, text: str, drain: bool=True) -> None: encoded = text.encode("utf-8", errors="ignore") await self._write_struct(msg, "L", len(encoded), drain=False) try: self.__writer.write(encoded) if drain: await self.__writer.drain() except ConnectionError as err: raise RfbConnectionError(f"Can't write {msg}", err) async def _write_fb_update(self, msg: str, width: int, height: int, encoding: int, drain: bool=True) -> None: await self._write_struct( msg, "BxH HH HH l", 0, # FB update 1, # Number of rects 0, 0, width, height, encoding, drain=drain, ) # ===== async def _start_tls(self, ssl_context: ssl.SSLContext, ssl_timeout: float) -> None: loop = asyncio.get_event_loop() ssl_reader = asyncio.StreamReader() protocol = asyncio.StreamReaderProtocol(ssl_reader) try: transport = await loop.start_tls( self.__writer.transport, protocol, ssl_context, server_side=True, ssl_handshake_timeout=ssl_timeout, ) except ConnectionError as err: raise RfbConnectionError("Can't start TLS", err) ssl_reader.set_transport(transport) ssl_writer = asyncio.StreamWriter( transport=transport, # type: ignore protocol=protocol, reader=ssl_reader, loop=loop, ) self.__reader = ssl_reader self.__writer = ssl_writer async def _close(self) -> None: await aiotools.close_writer(self.__writer)