diff options
Diffstat (limited to 'kvmd/plugins')
-rw-r--r-- | kvmd/plugins/atx/__init__.py | 82 | ||||
-rw-r--r-- | kvmd/plugins/atx/gpio.py | 181 | ||||
-rw-r--r-- | kvmd/plugins/atx/none.py | 78 |
3 files changed, 341 insertions, 0 deletions
diff --git a/kvmd/plugins/atx/__init__.py b/kvmd/plugins/atx/__init__.py new file mode 100644 index 00000000..77091c07 --- /dev/null +++ b/kvmd/plugins/atx/__init__.py @@ -0,0 +1,82 @@ +# ========================================================================== # +# # +# KVMD - The main Pi-KVM daemon. # +# # +# Copyright (C) 2018 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/>. # +# # +# ========================================================================== # + + +from typing import Dict +from typing import AsyncGenerator +from typing import Type + +from ... import aioregion + +from .. import BasePlugin +from .. import get_plugin_class + + +# ===== +class AtxError(Exception): + pass + + +class AtxOperationError(AtxError): + pass + + +class AtxIsBusyError(AtxOperationError, aioregion.RegionIsBusyError): + pass + + +# ===== +class BaseAtx(BasePlugin): + def get_state(self) -> Dict: + raise NotImplementedError + + async def poll_state(self) -> AsyncGenerator[Dict, None]: + yield {} + raise NotImplementedError + + async def power_on(self) -> bool: + raise NotImplementedError + + async def power_off(self) -> bool: + raise NotImplementedError + + async def power_off_hard(self) -> bool: + raise NotImplementedError + + async def power_reset_hard(self) -> bool: + raise NotImplementedError + + async def click_power(self) -> None: + raise NotImplementedError + + async def click_power_long(self) -> None: + raise NotImplementedError + + async def click_reset(self) -> None: + raise NotImplementedError + + async def cleanup(self) -> None: + pass + + +# ===== +def get_atx_class(name: str) -> Type[BaseAtx]: + return get_plugin_class("atx", (name or "none")) # type: ignore diff --git a/kvmd/plugins/atx/gpio.py b/kvmd/plugins/atx/gpio.py new file mode 100644 index 00000000..a7089a55 --- /dev/null +++ b/kvmd/plugins/atx/gpio.py @@ -0,0 +1,181 @@ +# ========================================================================== # +# # +# KVMD - The main Pi-KVM daemon. # +# # +# Copyright (C) 2018 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 operator + +from typing import Dict +from typing import AsyncGenerator + +from ...logging import get_logger + +from ... import aiotools +from ... import aioregion +from ... import gpio + +from ...yamlconf import Option + +from ...validators.basic import valid_bool +from ...validators.basic import valid_float_f01 + +from ...validators.hw import valid_gpio_pin + + +from . import AtxIsBusyError +from . import BaseAtx + + +# ===== +class Plugin(BaseAtx): # pylint: disable=too-many-instance-attributes + def __init__( # pylint: disable=too-many-arguments,super-init-not-called + self, + + power_led_pin: int, + hdd_led_pin: int, + power_led_inverted: bool, + hdd_led_inverted: bool, + + power_switch_pin: int, + reset_switch_pin: int, + click_delay: float, + long_click_delay: float, + + state_poll: float, + ) -> None: + + self.__power_led_pin = gpio.set_input(power_led_pin) + self.__hdd_led_pin = gpio.set_input(hdd_led_pin) + self.__power_switch_pin = gpio.set_output(power_switch_pin) + self.__reset_switch_pin = gpio.set_output(reset_switch_pin) + + self.__power_led_inverted = power_led_inverted + self.__hdd_led_inverted = hdd_led_inverted + + self.__click_delay = click_delay + self.__long_click_delay = long_click_delay + + self.__state_poll = state_poll + + self.__region = aioregion.AioExclusiveRegion(AtxIsBusyError) + + @classmethod + def get_plugin_options(cls) -> Dict[str, Option]: + return { + "power_led_pin": Option(-1, type=valid_gpio_pin), + "hdd_led_pin": Option(-1, type=valid_gpio_pin), + "power_led_inverted": Option(True, type=valid_bool), + "hdd_led_inverted": Option(True, type=valid_bool), + + "power_switch_pin": Option(-1, type=valid_gpio_pin), + "reset_switch_pin": Option(-1, type=valid_gpio_pin), + "click_delay": Option(0.1, type=valid_float_f01), + "long_click_delay": Option(5.5, type=valid_float_f01), + + "state_poll": Option(0.1, type=valid_float_f01), + } + + def get_state(self) -> Dict: + return { + "enabled": True, + "busy": self.__region.is_busy(), + "leds": { + "power": operator.xor(self.__power_led_inverted, gpio.read(self.__power_led_pin)), + "hdd": operator.xor(self.__hdd_led_inverted, gpio.read(self.__hdd_led_pin)), + }, + } + + async def poll_state(self) -> AsyncGenerator[Dict, None]: + prev_state: Dict = {} + while True: + state = self.get_state() + if state != prev_state: + yield state + prev_state = state + await asyncio.sleep(self.__state_poll) + + async def cleanup(self) -> None: + for (name, pin) in [ + ("power", self.__power_switch_pin), + ("reset", self.__reset_switch_pin), + ]: + try: + gpio.write(pin, False) + except Exception: + get_logger(0).exception("Can't cleanup %s pin %d", name, pin) + + # ===== + + async def power_on(self) -> bool: + if not self.get_state()["leds"]["power"]: + await self.click_power() + return True + return False + + async def power_off(self) -> bool: + if self.get_state()["leds"]["power"]: + await self.click_power() + return True + return False + + async def power_off_hard(self) -> bool: + if self.get_state()["leds"]["power"]: + await self.click_power_long() + return True + return False + + async def power_reset_hard(self) -> bool: + if self.get_state()["leds"]["power"]: + await self.click_reset() + return True + return False + + # ===== + + async def click_power(self) -> None: + await self.__click("power", self.__power_switch_pin, self.__click_delay) + + async def click_power_long(self) -> None: + await self.__click("power_long", self.__power_switch_pin, self.__long_click_delay) + + async def click_reset(self) -> None: + await self.__click("reset", self.__reset_switch_pin, self.__click_delay) + + # ===== + + @aiotools.atomic + async def __click(self, name: str, pin: int, delay: float) -> None: + with aiotools.unregion_only_on_exception(self.__region): + await self.__inner_click(name, pin, delay) + + @aiotools.tasked + @aiotools.muted("Can't perform ATX click or operation was not completed") + async def __inner_click(self, name: str, pin: int, delay: float) -> None: + try: + gpio.write(pin, True) + await asyncio.sleep(delay) + finally: + try: + gpio.write(pin, False) + await asyncio.sleep(1) + finally: + self.__region.exit() + get_logger(0).info("Clicked ATX button %r", name) diff --git a/kvmd/plugins/atx/none.py b/kvmd/plugins/atx/none.py new file mode 100644 index 00000000..09fad2f8 --- /dev/null +++ b/kvmd/plugins/atx/none.py @@ -0,0 +1,78 @@ +# ========================================================================== # +# # +# KVMD - The main Pi-KVM daemon. # +# # +# Copyright (C) 2018 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 + +from typing import Dict +from typing import AsyncGenerator + +from . import AtxOperationError +from . import BaseAtx + + +# ===== +class AtxDisabledError(AtxOperationError): + def __init__(self) -> None: + super().__init__("ATX is disabled") + + +# ===== +class Plugin(BaseAtx): + def get_state(self) -> Dict: + return { + "enabled": False, + "busy": False, + "leds": { + "power": False, + "hdd": False, + }, + } + + async def poll_state(self) -> AsyncGenerator[Dict, None]: + while True: + yield self.get_state() + await asyncio.sleep(60) + + # ===== + + async def power_on(self) -> bool: + raise AtxDisabledError() + + async def power_off(self) -> bool: + raise AtxDisabledError() + + async def power_off_hard(self) -> bool: + raise AtxDisabledError() + + async def power_reset_hard(self) -> bool: + raise AtxDisabledError() + + # ===== + + async def click_power(self) -> None: + raise AtxDisabledError() + + async def click_power_long(self) -> None: + raise AtxDisabledError() + + async def click_reset(self) -> None: + raise AtxDisabledError() |