diff options
author | Devaev Maxim <[email protected]> | 2020-09-05 09:26:55 +0300 |
---|---|---|
committer | Devaev Maxim <[email protected]> | 2020-09-05 09:26:55 +0300 |
commit | 482eeec3e715bbbd35dfafa4d8180d897fb549bd (patch) | |
tree | da3eca92d126e6ee549d4a14a5d342f426d2aa49 | |
parent | e162d84d5622f18a5afeddea0da19498ee25dc23 (diff) |
common background systasks
-rw-r--r-- | kvmd/apps/kvmd/server.py | 5 | ||||
-rw-r--r-- | kvmd/apps/kvmd/ugpio.py | 31 | ||||
-rw-r--r-- | kvmd/plugins/msd/otg/__init__.py | 31 |
3 files changed, 22 insertions, 45 deletions
diff --git a/kvmd/apps/kvmd/server.py b/kvmd/apps/kvmd/server.py index 9fdf445f..65c10d47 100644 --- a/kvmd/apps/kvmd/server.py +++ b/kvmd/apps/kvmd/server.py @@ -112,13 +112,14 @@ class _Component: obj: object get_state: Optional[Callable[[], Coroutine[Any, Any, Dict]]] = None poll_state: Optional[Callable[[], AsyncGenerator[Dict, None]]] = None + systask: Optional[Callable[[], Coroutine[Any, Any, None]]] = None cleanup: Optional[Callable[[], Coroutine[Any, Any, Dict]]] = None def __post_init__(self) -> None: if isinstance(self.obj, BasePlugin): object.__setattr__(self, "name", f"{self.name} ({self.obj.get_plugin_name()})") - for field in ["get_state", "poll_state", "cleanup"]: + for field in ["get_state", "poll_state", "systask", "cleanup"]: object.__setattr__(self, field, getattr(self.obj, field, None)) if self.get_state or self.poll_state: assert self.event_type, self @@ -288,6 +289,8 @@ class KvmdServer(HttpServer): # pylint: disable=too-many-arguments,too-many-ins self.__run_system_task(self.__stream_controller) for component in self.__components: + if component.systask: + self.__run_system_task(component.systask) if component.poll_state: self.__run_system_task(self.__poll_state, component.event_type, component.poll_state()) self.__run_system_task(self.__stream_snapshoter) diff --git a/kvmd/apps/kvmd/ugpio.py b/kvmd/apps/kvmd/ugpio.py index 81c191d7..b5124f78 100644 --- a/kvmd/apps/kvmd/ugpio.py +++ b/kvmd/apps/kvmd/ugpio.py @@ -201,29 +201,16 @@ class UserGpio: } async def poll_state(self) -> AsyncGenerator[Dict, None]: - reader_task = asyncio.create_task(self.__reader.poll()) - waiter_task: Optional[asyncio.Task] = None prev_state: Dict = {} - try: - while True: - if reader_task.cancelled(): - break - if reader_task.done(): - RuntimeError("BatchReader task is dead") - - state = await self.get_state() - if state != prev_state: - yield state - prev_state = state - - if waiter_task is None: - waiter_task = asyncio.create_task(self.__state_notifier.wait()) - if waiter_task in (await aiotools.wait_first(reader_task, waiter_task))[0]: - waiter_task = None - finally: - if not reader_task.done(): - reader_task.cancel() - await reader_task + while True: + state = await self.get_state() + if state != prev_state: + yield state + prev_state = state + await self.__state_notifier.wait() + + async def systask(self) -> None: + await self.__reader.poll() async def cleanup(self) -> None: for gout in self.__outputs.values(): diff --git a/kvmd/plugins/msd/otg/__init__.py b/kvmd/plugins/msd/otg/__init__.py index 25585302..8d1cc1b2 100644 --- a/kvmd/plugins/msd/otg/__init__.py +++ b/kvmd/plugins/msd/otg/__init__.py @@ -206,29 +206,16 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes } async def poll_state(self) -> AsyncGenerator[Dict, None]: - inotify_task = asyncio.create_task(self.__watch_inotify()) - waiter_task: Optional[asyncio.Task] = None prev_state: Dict = {} - try: - while True: - if inotify_task.cancelled(): - break - if inotify_task.done(): - RuntimeError("Inotify task is dead") - - state = await self.get_state() - if state != prev_state: - yield state - prev_state = state - - if waiter_task is None: - waiter_task = asyncio.create_task(self.__state_notifier.wait()) - if waiter_task in (await aiotools.wait_first(inotify_task, waiter_task))[0]: - waiter_task = None - finally: - if not inotify_task.done(): - inotify_task.cancel() - await inotify_task + while True: + state = await self.get_state() + if state != prev_state: + yield state + prev_state = state + await self.__state_notifier.wait() + + async def systask(self) -> None: + await self.__watch_inotify() @aiotools.atomic async def reset(self) -> None: |