diff options
Diffstat (limited to 'kvmd/plugins/msd/relay/__init__.py')
-rw-r--r-- | kvmd/plugins/msd/relay/__init__.py | 293 |
1 files changed, 0 insertions, 293 deletions
diff --git a/kvmd/plugins/msd/relay/__init__.py b/kvmd/plugins/msd/relay/__init__.py deleted file mode 100644 index a93d5e34..00000000 --- a/kvmd/plugins/msd/relay/__init__.py +++ /dev/null @@ -1,293 +0,0 @@ -# ========================================================================== # -# # -# KVMD - The main PiKVM daemon. # -# # -# Copyright (C) 2018-2022 Maxim Devaev <[email protected]> # -# # -# 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 <https://www.gnu.org/licenses/>. # -# # -# ========================================================================== # - - -import asyncio -import contextlib -import dataclasses -import functools - -from typing import AsyncGenerator - -from ....logging import get_logger - -from .... import aiotools - -from ....yamlconf import Option - -from ....validators.basic import valid_bool -from ....validators.basic import valid_number -from ....validators.basic import valid_int_f1 -from ....validators.basic import valid_float_f01 -from ....validators.os import valid_abs_path -from ....validators.hw import valid_gpio_pin - -from .. import MsdError -from .. import MsdIsBusyError -from .. import MsdOfflineError -from .. import MsdConnectedError -from .. import MsdDisconnectedError -from .. import MsdMultiNotSupported -from .. import MsdCdromNotSupported -from .. import MsdRwNotSupported -from .. import BaseMsdReader -from .. import BaseMsd -from .. import MsdFileWriter - -from .gpio import Gpio - -from .drive import DeviceInfo - - -# ===== -class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes - def __init__( # pylint: disable=super-init-not-called,too-many-arguments - self, - upload_chunk_size: int, - sync_chunk_size: int, - - gpio_device_path: str, - target_pin: int, - reset_inverted: bool, - reset_pin: int, - - device_path: str, - init_delay: float, - init_retries: int, - reset_delay: float, - ) -> None: - - self.__upload_chunk_size = upload_chunk_size - self.__sync_chunk_size = sync_chunk_size - - self.__device_path = device_path - self.__init_delay = init_delay - self.__init_retries = init_retries - - self.__gpio = Gpio(gpio_device_path, target_pin, reset_pin, reset_inverted, reset_delay) - - self.__device_info: (DeviceInfo | None) = None - self.__connected = False - - self.__device_writer: (MsdFileWriter | None) = None - - self.__notifier = aiotools.AioNotifier() - self.__region = aiotools.AioExclusiveRegion(MsdIsBusyError, self.__notifier) - - @classmethod - def get_plugin_options(cls) -> dict: - return { - "upload_chunk_size": Option(65536, type=functools.partial(valid_number, min=1024)), - "sync_chunk_size": Option(4194304, type=functools.partial(valid_number, min=1024)), - - "gpio_device": Option("/dev/gpiochip0", type=valid_abs_path, unpack_as="gpio_device_path"), - "target_pin": Option(-1, type=valid_gpio_pin), - "reset_pin": Option(-1, type=valid_gpio_pin), - "reset_inverted": Option(False, type=valid_bool), - - "device": Option("", type=valid_abs_path, unpack_as="device_path"), - "init_delay": Option(1.0, type=valid_float_f01), - "init_retries": Option(5, type=valid_int_f1), - "reset_delay": Option(1.0, type=valid_float_f01), - } - - def sysprep(self) -> None: - logger = get_logger(0) - self.__gpio.open() - logger.info("Using %r as MSD", self.__device_path) - try: - aiotools.run_sync(self.__load_device_info()) - except Exception as err: - log = (logger.error if isinstance(err, MsdError) else logger.exception) - log("MSD is offline: %s", err) - - async def get_state(self) -> dict: - storage: (dict | None) = None - drive: (dict | None) = None - if self.__device_info: - storage = { - "size": self.__device_info.size, - "free": self.__device_info.free, - "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)), - "connected": self.__connected, - } - return { - "enabled": True, - "online": bool(self.__device_info), - "busy": self.__region.is_busy(), - "storage": storage, - "drive": drive, - "features": { - "multi": False, - "cdrom": False, - "rw": False, - }, - } - - async def poll_state(self) -> AsyncGenerator[dict, None]: - prev_state: dict = {} - while True: - state = await self.get_state() - if state != prev_state: - yield state - prev_state = state - await self.__notifier.wait() - - @aiotools.atomic_fg - async def reset(self) -> None: - await aiotools.run_region_task( - "Can't reset MSD or operation was not completed", - self.__region, self.__inner_reset, - ) - - @aiotools.atomic_fg - async def __inner_reset(self) -> None: - await self.__gpio.reset() - self.__gpio.switch_to_local() - self.__connected = False - await self.__load_device_info() - get_logger(0).info("MSD reset has been successful") - - @aiotools.atomic_fg - async def cleanup(self) -> None: - try: - await self.__close_device_writer() - finally: - self.__gpio.close() - - # ===== - - @aiotools.atomic_fg - async def set_params( - self, - name: (str | None)=None, - cdrom: (bool | None)=None, - rw: (bool | None)=None, - ) -> None: - - async with self.__working(): - if name is not None: - raise MsdMultiNotSupported() - if cdrom is not None: - raise MsdCdromNotSupported() - if rw is not None: - raise MsdRwNotSupported() - - @aiotools.atomic_fg - async def set_connected(self, connected: bool) -> None: - async with self.__working(): - async with self.__region: - if connected: - if self.__connected: - raise MsdConnectedError() - self.__gpio.switch_to_server() - get_logger(0).info("MSD switched to Server") - else: - if not self.__connected: - raise MsdDisconnectedError() - self.__gpio.switch_to_local() - try: - await self.__load_device_info() - except Exception: - if self.__connected: - self.__gpio.switch_to_server() - raise - get_logger(0).info("MSD switched to KVM: %s", self.__device_info) - self.__connected = connected - - @contextlib.asynccontextmanager - async def read_image(self, name: str) -> AsyncGenerator[BaseMsdReader, None]: - async with self.__working(): - if self is not None: # XXX: Vulture and pylint hack - raise MsdMultiNotSupported() - yield BaseMsdReader() - - @contextlib.asynccontextmanager - async def write_image(self, name: str, size: int, remove_incomplete: (bool | None)) -> AsyncGenerator[MsdFileWriter, None]: - async with self.__working(): - if remove_incomplete is not None: - raise MsdMultiNotSupported() - async with self.__region: - try: - assert self.__device_info - if self.__connected: - raise MsdConnectedError() - - self.__device_writer = await MsdFileWriter( - notifier=self.__notifier, - path=self.__device_info.path, - file_size=size, - sync_size=self.__sync_chunk_size, - chunk_size=self.__upload_chunk_size, - ).open() - - await self.__write_image_info(False) - self.__notifier.notify() - yield self.__device_writer - await self.__write_image_info(True) - finally: - try: - await aiotools.shield_fg(self.__close_device_writer()) - finally: - await aiotools.shield_fg(self.__load_device_info()) - - @aiotools.atomic_fg - async def remove(self, name: str) -> None: - async with self.__working(): - raise MsdMultiNotSupported() - - # ===== - - @contextlib.asynccontextmanager - async def __working(self) -> AsyncGenerator[None, None]: - if not self.__device_info: - raise MsdOfflineError() - yield - - # ===== - - 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(self.__device_writer, complete)): - get_logger().error("Can't write image info because device is full") - - async def __close_device_writer(self) -> None: - if self.__device_writer: - await self.__device_writer.close() # type: ignore - self.__device_writer = None - - async def __load_device_info(self) -> None: - retries = self.__init_retries - while True: - await asyncio.sleep(self.__init_delay) - try: - self.__device_info = await DeviceInfo.read(self.__device_path) - break - except Exception: - if retries == 0: - self.__device_info = None - raise MsdError("Can't load device info") - get_logger().exception("Can't load device info; retries=%d", retries) - retries -= 1 |