diff options
Diffstat (limited to 'kvmd/plugins/msd/relay/__init__.py')
-rw-r--r-- | kvmd/plugins/msd/relay/__init__.py | 56 |
1 files changed, 18 insertions, 38 deletions
diff --git a/kvmd/plugins/msd/relay/__init__.py b/kvmd/plugins/msd/relay/__init__.py index 659594d8..82de6596 100644 --- a/kvmd/plugins/msd/relay/__init__.py +++ b/kvmd/plugins/msd/relay/__init__.py @@ -29,13 +29,9 @@ from typing import Dict from typing import AsyncGenerator from typing import Optional -import aiofiles -import aiofiles.base - from ....logging import get_logger from .... import aiotools -from .... import aiofs from ....yamlconf import Option @@ -54,10 +50,10 @@ from .. import MsdDisconnectedError from .. import MsdMultiNotSupported from .. import MsdCdromNotSupported from .. import BaseMsd +from .. import MsdImageWriter from .gpio import Gpio -from .drive import ImageInfo from .drive import DeviceInfo @@ -91,9 +87,7 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes self.__device_info: Optional[DeviceInfo] = None self.__connected = False - self.__device_file: Optional[aiofiles.base.AiofilesContextManager] = None - self.__written = 0 - self.__unsynced = 0 + self.__device_writer: Optional[MsdImageWriter] = None self.__notifier = aiotools.AioNotifier() self.__region = aiotools.AioExclusiveRegion(MsdIsBusyError, self.__notifier) @@ -132,7 +126,7 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes storage = { "size": self.__device_info.size, "free": self.__device_info.free, - "uploading": bool(self.__device_file) + "uploading": (self.__device_writer.get_state() if self.__device_writer else None), } drive = { "image": (self.__device_info.image and dataclasses.asdict(self.__device_info.image)), @@ -177,7 +171,7 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes @aiotools.atomic async def cleanup(self) -> None: try: - await self.__close_device_file() + await self.__close_device_writer() finally: self.__gpio.close() @@ -214,7 +208,7 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes self.__connected = connected @contextlib.asynccontextmanager - async def write_image(self, name: str) -> AsyncGenerator[None, None]: + async def write_image(self, name: str, size: int) -> AsyncGenerator[None, None]: async with self.__working(): async with self.__region: try: @@ -222,30 +216,22 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes if self.__connected: raise MsdConnectedError() - self.__device_file = await aiofiles.open(self.__device_info.path, mode="w+b", buffering=0) # type: ignore - self.__written = 0 - self.__unsynced = 0 + self.__device_writer = await MsdImageWriter(self.__device_info.path, size, self.__sync_chunk_size).open() - await self.__write_image_info(name, complete=False) + await self.__write_image_info(False) await self.__notifier.notify() yield - await self.__write_image_info(name, complete=True) + await self.__write_image_info(True) finally: - await self.__close_device_file() + await self.__close_device_writer() await self.__load_device_info() def get_upload_chunk_size(self) -> int: return self.__upload_chunk_size async def write_image_chunk(self, chunk: bytes) -> int: - assert self.__device_file - await self.__device_file.write(chunk) # type: ignore - self.__written += len(chunk) - self.__unsynced += len(chunk) - if self.__unsynced >= self.__sync_chunk_size: - await aiofs.afile_sync(self.__device_file) - self.__unsynced = 0 - return self.__written + assert self.__device_writer + return (await self.__device_writer.write(chunk)) @aiotools.atomic async def remove(self, name: str) -> None: @@ -262,27 +248,21 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes # ===== - async def __write_image_info(self, name: str, complete: bool) -> None: - assert self.__device_file + async def __write_image_info(self, complete: bool) -> None: + assert self.__device_writer assert self.__device_info - if not (await self.__device_info.write_image_info( - device_file=self.__device_file, - image_info=ImageInfo(name, self.__written, complete), - )): + if not (await self.__device_info.write_image_info(self.__device_writer, complete)): get_logger().error("Can't write image info because device is full") - async def __close_device_file(self) -> None: + async def __close_device_writer(self) -> None: try: - if self.__device_file: + if self.__device_writer: get_logger().info("Closing device file ...") - await aiofs.afile_sync(self.__device_file) - await self.__device_file.close() # type: ignore + await self.__device_writer.close() # type: ignore except Exception: get_logger().exception("Can't close device file") finally: - self.__device_file = None - self.__written = 0 - self.__unsynced = 0 + self.__device_writer = None async def __load_device_info(self) -> None: retries = self.__init_retries |