diff options
-rw-r--r-- | kvmd/PKGBUILD | 2 | ||||
-rw-r--r-- | kvmd/dev_requirements.txt | 2 | ||||
-rw-r--r-- | kvmd/kvmd/__init__.py | 25 | ||||
-rw-r--r-- | kvmd/kvmd/extras/exploremsd/__init__.py | 20 | ||||
-rw-r--r-- | kvmd/kvmd/extras/exploremsd/__main__.py | 2 | ||||
-rw-r--r-- | kvmd/kvmd/msd.py | 197 | ||||
-rw-r--r-- | kvmd/kvmd/ps2.py | 4 | ||||
-rw-r--r-- | kvmd/kvmd/server.py | 81 | ||||
-rw-r--r-- | kvmd/kvmd/streamer.py | 4 | ||||
-rw-r--r-- | kvmd/requirements.txt | 2 | ||||
-rwxr-xr-x | kvmd/setup.py | 2 | ||||
-rw-r--r-- | os/platforms/v1/kvmd.yaml | 7 |
12 files changed, 332 insertions, 16 deletions
diff --git a/kvmd/PKGBUILD b/kvmd/PKGBUILD index a46b3093..9d97eda5 100644 --- a/kvmd/PKGBUILD +++ b/kvmd/PKGBUILD @@ -13,6 +13,8 @@ depends=( "python" "python-yaml" "python-aiohttp" + "python-aiofiles" + "python-pyudev" "python-raspberry-gpio" ) makedepends=("python-setuptools" "wget") diff --git a/kvmd/dev_requirements.txt b/kvmd/dev_requirements.txt index 4e3e1940..d054c0ee 100644 --- a/kvmd/dev_requirements.txt +++ b/kvmd/dev_requirements.txt @@ -1,5 +1,7 @@ git+git://github.com/willbuckner/rpi-gpio-development-mock@master#egg=rpi aiohttp +aiofiles +pyudev pyyaml bumpversion tox diff --git a/kvmd/kvmd/__init__.py b/kvmd/kvmd/__init__.py index ed8ce8da..8b950212 100644 --- a/kvmd/kvmd/__init__.py +++ b/kvmd/kvmd/__init__.py @@ -3,9 +3,10 @@ import asyncio from .application import init from .logging import get_logger +from .ps2 import Ps2Keyboard from .atx import Atx +from .msd import MassStorageDevice from .streamer import Streamer -from .ps2 import Ps2Keyboard from .server import Server from . import gpio @@ -17,6 +18,12 @@ def main() -> None: with gpio.bcm(): loop = asyncio.get_event_loop() + keyboard = Ps2Keyboard( + clock=int(config["keyboard"]["pinout"]["clock"]), + data=int(config["keyboard"]["pinout"]["data"]), + pulse=float(config["keyboard"]["pulse"]), + ) + atx = Atx( power_led=int(config["atx"]["leds"]["pinout"]["power"]), hdd_led=int(config["atx"]["leds"]["pinout"]["hdd"]), @@ -26,6 +33,12 @@ def main() -> None: long_click_delay=float(config["atx"]["switches"]["long_click_delay"]), ) + msd = MassStorageDevice( + bind=str(config["msd"]["bind"]), + init_delay=float(config["msd"]["init_delay"]), + loop=loop, + ) + streamer = Streamer( cap_power=int(config["video"]["pinout"]["cap"]), conv_power=int(config["video"]["pinout"]["conv"]), @@ -34,19 +47,15 @@ def main() -> None: loop=loop, ) - keyboard = Ps2Keyboard( - clock=int(config["keyboard"]["pinout"]["clock"]), - data=int(config["keyboard"]["pinout"]["data"]), - pulse=float(config["keyboard"]["pulse"]), - ) - Server( + keyboard=keyboard, atx=atx, + msd=msd, streamer=streamer, - keyboard=keyboard, heartbeat=float(config["server"]["heartbeat"]), atx_leds_poll=float(config["atx"]["leds"]["poll"]), video_shutdown_delay=float(config["video"]["shutdown_delay"]), + msd_chunk_size=int(config["msd"]["chunk_size"]), loop=loop, ).run( host=str(config["server"]["host"]), diff --git a/kvmd/kvmd/extras/exploremsd/__init__.py b/kvmd/kvmd/extras/exploremsd/__init__.py new file mode 100644 index 00000000..e3680f6c --- /dev/null +++ b/kvmd/kvmd/extras/exploremsd/__init__.py @@ -0,0 +1,20 @@ +import argparse + +from ...msd import explore_device +from ...msd import locate_by_bind + + +# ===== +def main() -> None: + parser = argparse.ArgumentParser() + parser.add_argument("-d", "--device", default="/dev/sda") + options = parser.parse_args() + + info = explore_device(options.device) + print("Path:", info.path) + print("Bind:", info.bind) + print("Size:", info.size) + print("Manufacturer:", info.manufacturer) + print("Product:", info.product) + print("Serial:", info.serial) + assert locate_by_bind(info.bind), "WTF?! Can't locate device file using bind %r" % (info.bind) diff --git a/kvmd/kvmd/extras/exploremsd/__main__.py b/kvmd/kvmd/extras/exploremsd/__main__.py new file mode 100644 index 00000000..031df43e --- /dev/null +++ b/kvmd/kvmd/extras/exploremsd/__main__.py @@ -0,0 +1,2 @@ +from . import main +main() diff --git a/kvmd/kvmd/msd.py b/kvmd/kvmd/msd.py new file mode 100644 index 00000000..98ce07bc --- /dev/null +++ b/kvmd/kvmd/msd.py @@ -0,0 +1,197 @@ +import os +import asyncio +import types + +from typing import Dict +from typing import NamedTuple +from typing import Callable +from typing import Type +from typing import Optional +from typing import Any + +import pyudev + +import aiofiles +import aiofiles.base + +from .logging import get_logger + + +# ===== +class MassStorageError(Exception): + pass + + +class MassStorageIsNotOperationalError(MassStorageError): + def __init__(self) -> None: + super().__init__("Missing bind for mass-storage device") + + +class MassStorageAlreadyConnectedToPcError(MassStorageError): + def __init__(self) -> None: + super().__init__("Mass-storage is already connected to PC") + + +class MassStorageAlreadyConnectedToKvmError(MassStorageError): + def __init__(self) -> None: + super().__init__("Mass-storage is already connected to KVM") + + +class MassStorageIsNotConnectedToKvmError(MassStorageError): + def __init__(self) -> None: + super().__init__("Mass-storage is not connected to KVM") + + +class MassStorageIsBusyError(MassStorageError): + def __init__(self) -> None: + super().__init__("Mass-storage is busy (write in progress)") + + +class DeviceInfo(NamedTuple): + path: str + bind: str + size: int + manufacturer: str + product: str + serial: str + + +def explore_device(path: str) -> DeviceInfo: + # udevadm info -a -p $(udevadm info -q path -n /dev/sda) + ctx = pyudev.Context() + + block_device = pyudev.Devices.from_device_file(ctx, path) + size = block_device.attributes.asint("size") * 512 + + storage_device = block_device.find_parent("usb", "usb_interface") + assert storage_device.driver == "usb-storage", (storage_device.driver, storage_device) + + usb_device = block_device.find_parent("usb", "usb_device") + assert usb_device.driver == "usb", (usb_device.driver, usb_device) + + return DeviceInfo( + path=path, + bind=storage_device.sys_name, + size=size, + manufacturer=usb_device.attributes.asstring("manufacturer").strip(), + product=usb_device.attributes.asstring("product").strip(), + serial=usb_device.attributes.asstring("serial").strip(), + ) + + +def locate_by_bind(bind: str) -> str: + ctx = pyudev.Context() + for device in ctx.list_devices(subsystem="block"): + storage_device = device.find_parent("usb", "usb_interface") + if storage_device: + try: + device.attributes.asint("partititon") + except KeyError: + if storage_device.sys_name == bind: + return os.path.join("/dev", device.sys_name) + return "" + + +def _operated_and_locked(method: Callable) -> Callable: + async def wrap(self: "MassStorageDevice", *args: Any, **kwargs: Any) -> Any: + if self._device_file: # pylint: disable=protected-access + raise MassStorageIsBusyError() + if not self._bind: # pylint: disable=protected-access + MassStorageIsNotOperationalError() + async with self._lock: # pylint: disable=protected-access + return (await method(self, *args, **kwargs)) + return wrap + + +class MassStorageDevice: + def __init__(self, bind: str, init_delay: float, loop: asyncio.AbstractEventLoop) -> None: + self._bind = bind + self.__init_delay = init_delay + + self.__device_info: Optional[DeviceInfo] = None + self._lock = asyncio.Lock() + self._device_file: Optional[aiofiles.base.AiofilesContextManager] = None + self.__writed = 0 + + if self._bind: + get_logger().info("Using bind %r as mass-storage device", self._bind) + try: + loop.run_until_complete(self.connect_to_kvm(no_delay=True)) + except Exception: + get_logger().exception("Mass-storage device is not operational") + self._bind = "" + else: + get_logger().warning("Missing bind; mass-storage device is not operational") + + @_operated_and_locked + async def connect_to_kvm(self, no_delay: bool=False) -> None: + if self.__device_info: + raise MassStorageAlreadyConnectedToKvmError() + # TODO: disable gpio + if not no_delay: + await asyncio.sleep(self.__init_delay) + path = locate_by_bind(self._bind) + if not path: + raise RuntimeError("Can't locate device by bind %r" % (self._bind)) + self.__device_info = explore_device(path) + get_logger().info("Mass-storage device switched to KVM: %s", self.__device_info) + + @_operated_and_locked + async def connect_to_pc(self) -> None: + if not self.__device_info: + raise MassStorageAlreadyConnectedToPcError() + # TODO: enable gpio + self.__device_info = None + get_logger().info("Mass-storage device switched to PC") + + def get_state(self) -> Dict: + return { + "in_operate": bool(self._bind), + "connected_to": ("kvm" if self.__device_info else "pc"), + "is_busy": bool(self._device_file), + "writed": self.__writed, + "info": (self.__device_info._asdict() if self.__device_info else None), + } + + async def cleanup(self) -> None: + async with self._lock: + await self.__close_file() + # TODO: disable gpio + + @_operated_and_locked + async def __aenter__(self) -> "MassStorageDevice": + if not self.__device_info: + raise MassStorageIsNotConnectedToKvmError() + self._device_file = await aiofiles.open(self.__device_info.path, mode="wb", buffering=0) + self.__writed = 0 + return self + + async def write(self, data: bytes) -> int: + async with self._lock: + assert self._device_file + size = len(data) + await self._device_file.write(data) + await self._device_file.flush() + os.fsync(self._device_file.fileno()) + self.__writed += size + return self.__writed + + async def __aexit__( + self, + _exc_type: Type[BaseException], + _exc: BaseException, + _tb: types.TracebackType, + ) -> None: + async with self._lock: + await self.__close_file() + + async def __close_file(self) -> None: + try: + if self._device_file: + get_logger().info("Closing device file ...") + await self._device_file.close() + except Exception: + get_logger().exception("Can't close device file") + # TODO: reset device file + self._device_file = None + self.__writed = 0 diff --git a/kvmd/kvmd/ps2.py b/kvmd/kvmd/ps2.py index 7711ad6b..7f6bc85d 100644 --- a/kvmd/kvmd/ps2.py +++ b/kvmd/kvmd/ps2.py @@ -32,6 +32,10 @@ class Ps2Keyboard(multiprocessing.Process): def send_byte(self, code: int) -> None: self.__queue.put(code) + def cleanup(self) -> None: + if self.is_alive(): + self.stop() + def run(self) -> None: with gpio.bcm(): try: diff --git a/kvmd/kvmd/server.py b/kvmd/kvmd/server.py index 7d87a955..0a9e848a 100644 --- a/kvmd/kvmd/server.py +++ b/kvmd/kvmd/server.py @@ -7,12 +7,18 @@ from typing import List from typing import Set from typing import Callable from typing import Optional +from typing import Type import aiohttp.web +from .ps2 import Ps2Keyboard + from .atx import Atx + +from .msd import MassStorageError +from .msd import MassStorageDevice + from .streamer import Streamer -from .ps2 import Ps2Keyboard from .logging import get_logger @@ -30,24 +36,43 @@ def _system_task(method: Callable) -> Callable: return wrap +def _exceptions_as_400(msg: str, exceptions: List[Type[Exception]]) -> Callable: + def make_wrapper(method: Callable) -> Callable: + async def wrap(self: "Server", request: aiohttp.web.Request) -> aiohttp.web.WebSocketResponse: + try: + return (await method(self, request)) + except tuple(exceptions) as err: # pylint: disable=catching-non-exception + get_logger().exception(msg) + return aiohttp.web.json_response({ + "error": type(err).__name__, + "error_msg": str(err), + }, status=400) + return wrap + return make_wrapper + + class Server: # pylint: disable=too-many-instance-attributes def __init__( self, + keyboard: Ps2Keyboard, atx: Atx, + msd: MassStorageDevice, streamer: Streamer, - keyboard: Ps2Keyboard, heartbeat: float, atx_leds_poll: float, video_shutdown_delay: float, + msd_chunk_size: int, loop: asyncio.AbstractEventLoop, ) -> None: + self.__keyboard = keyboard self.__atx = atx + self.__msd = msd self.__streamer = streamer self.__heartbeat = heartbeat - self.__keyboard = keyboard self.__video_shutdown_delay = video_shutdown_delay self.__atx_leds_poll = atx_leds_poll + self.__msd_chunk_size = msd_chunk_size self.__loop = loop self.__sockets: Set[aiohttp.web.WebSocketResponse] = set() @@ -63,6 +88,9 @@ class Server: # pylint: disable=too-many-instance-attributes app = aiohttp.web.Application(loop=self.__loop) app.router.add_get("/", self.__root_handler) app.router.add_get("/ws", self.__ws_handler) + app.router.add_get("/msd", self.__msd_state_handler) + app.router.add_post("/msd/connect", self.__msd_connect_handler) + app.router.add_post("/msd/write", self.__msd_write_handler) app.on_shutdown.append(self.__on_shutdown) app.on_cleanup.append(self.__on_cleanup) @@ -91,6 +119,46 @@ class Server: # pylint: disable=too-many-instance-attributes break return ws + async def __msd_state_handler(self, _: aiohttp.web.Request) -> aiohttp.web.Response: + return aiohttp.web.json_response(self.__msd.get_state()) + + @_exceptions_as_400("Mass-storage error", [MassStorageError, RuntimeError]) + async def __msd_connect_handler(self, request: aiohttp.web.Request) -> aiohttp.web.Response: + to = request.query.get("to") + if to == "kvm": + await self.__msd.connect_to_kvm() + await self.__broadcast("EVENT msd connected_to_kvm") + elif to == "pc": + await self.__msd.connect_to_pc() + await self.__broadcast("EVENT msd connected_to_pc") + else: + raise RuntimeError("Missing or invalid 'to=%s'" % (to)) + return aiohttp.web.json_response(self.__msd.get_state()) + + @_exceptions_as_400("Can't write image to mass-storage device", [MassStorageError, RuntimeError, OSError]) + async def __msd_write_handler(self, request: aiohttp.web.Request) -> aiohttp.web.Response: + logger = get_logger(0) + reader = await request.multipart() + writed = 0 + try: + field = await reader.next() + if field.name != "image": + raise RuntimeError("Missing 'data' field") + + async with self.__msd: + await self.__broadcast("EVENT msd busy") + logger.info("Writing image to mass-storage device ...") + while True: + chunk = await field.read_chunk(self.__msd_chunk_size) + if not chunk: + break + writed = await self.__msd.write(chunk) + await self.__broadcast("EVENT msd free") + finally: + if writed != 0: + logger.info("Writed %d bytes to mass-storage device", writed) + return aiohttp.web.json_response({"writed": writed}) + def __run_app_print(self, text: str) -> None: logger = get_logger() for line in text.strip().splitlines(): @@ -109,10 +177,9 @@ class Server: # pylint: disable=too-many-instance-attributes await self.__remove_socket(ws) async def __on_cleanup(self, _: aiohttp.web.Application) -> None: - if self.__keyboard.is_alive(): - self.__keyboard.stop() - if self.__streamer.is_running(): - await self.__streamer.stop() + self.__keyboard.cleanup() + await self.__streamer.cleanup() + await self.__msd.cleanup() @_system_task async def __keyboard_watchdog(self) -> None: diff --git a/kvmd/kvmd/streamer.py b/kvmd/kvmd/streamer.py index 8cb5a486..72f818cb 100644 --- a/kvmd/kvmd/streamer.py +++ b/kvmd/kvmd/streamer.py @@ -48,6 +48,10 @@ class Streamer: # pylint: disable=too-many-instance-attributes def is_running(self) -> bool: return bool(self.__proc_task) + async def cleanup(self) -> None: + if self.is_running(): + await self.stop() + async def __set_hw_enabled(self, enabled: bool) -> None: # XXX: This sequence is very important to enable converter and cap board gpio.write(self.__cap_power, enabled) diff --git a/kvmd/requirements.txt b/kvmd/requirements.txt index f688298d..bc7501dc 100644 --- a/kvmd/requirements.txt +++ b/kvmd/requirements.txt @@ -1,3 +1,5 @@ RPi.GPIO aiohttp +aiofiles +pyudev pyyaml diff --git a/kvmd/setup.py b/kvmd/setup.py index 4d8f5f10..502c7003 100755 --- a/kvmd/setup.py +++ b/kvmd/setup.py @@ -24,6 +24,7 @@ def main() -> None: "kvmd.extras", "kvmd.extras.cleanup", "kvmd.extras.wscli", + "kvmd.extras.exploremsd", ], entry_points={ @@ -31,6 +32,7 @@ def main() -> None: "kvmd = kvmd:main", "kvmd-cleanup = kvmd.extras.cleanup:main", "kvmd-wscli = kvmd.extras.wscli:main", + "kvmd-exploremsd = kvmd.extras.exploremsd:main", ], }, diff --git a/os/platforms/v1/kvmd.yaml b/os/platforms/v1/kvmd.yaml index 3dcc8d04..833c730e 100644 --- a/os/platforms/v1/kvmd.yaml +++ b/os/platforms/v1/kvmd.yaml @@ -24,12 +24,17 @@ kvmd: click_delay: 0.1 long_click_delay: 5.5 + msd: + # FIXME: It's for laptop lol + bind: "1-2:1.0" + init_delay: 2.0 + chunk_size: 512 + video: pinout: cap: 21 conv: 25 sync_delay: 1.0 - shutdown_delay: 10.0 cmd: |