summaryrefslogtreecommitdiff
path: root/kvmd
diff options
context:
space:
mode:
Diffstat (limited to 'kvmd')
-rw-r--r--kvmd/inotify.py6
-rw-r--r--kvmd/plugins/msd/otg/__init__.py10
-rw-r--r--kvmd/plugins/msd/otg/storage.py50
-rw-r--r--kvmd/plugins/ugpio/otgconf.py4
4 files changed, 46 insertions, 24 deletions
diff --git a/kvmd/inotify.py b/kvmd/inotify.py
index d76e73e3..fe9b6391 100644
--- a/kvmd/inotify.py
+++ b/kvmd/inotify.py
@@ -34,6 +34,7 @@ from typing import Generator
from .logging import get_logger
+from . import aiotools
from . import libc
@@ -189,11 +190,12 @@ class Inotify:
self.__events_queue: "asyncio.Queue[InotifyEvent]" = asyncio.Queue()
- def watch(self, path: str, mask: int) -> None:
+ async def watch(self, path: str, mask: int) -> None:
path = os.path.normpath(path)
assert path not in self.__wd_by_path, path
get_logger().info("Watching for %s", path)
- wd = _inotify_check(libc.inotify_add_watch(self.__fd, _fs_encode(path), mask))
+ # Асинхронно, чтобы не висло на NFS
+ wd = _inotify_check(await aiotools.run_async(libc.inotify_add_watch, self.__fd, _fs_encode(path), mask))
self.__wd_by_path[path] = wd
self.__path_by_wd[wd] = path
diff --git a/kvmd/plugins/msd/otg/__init__.py b/kvmd/plugins/msd/otg/__init__.py
index ddb90a79..53082cc6 100644
--- a/kvmd/plugins/msd/otg/__init__.py
+++ b/kvmd/plugins/msd/otg/__init__.py
@@ -431,10 +431,10 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
with Inotify() as inotify:
for path in [
- *self.__storage.get_watchable_paths(),
+ *(await self.__storage.get_watchable_paths()),
*self.__drive.get_watchable_paths(),
]:
- inotify.watch(path, InotifyMask.ALL_MODIFY_EVENTS)
+ await inotify.watch(path, InotifyMask.ALL_MODIFY_EVENTS)
# После установки вотчеров еще раз проверяем стейт, чтобы ничего не потерять
await self.__reload_state()
@@ -471,7 +471,7 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
await self.__storage.remount_rw(False)
await self.__setup_initial()
- storage_state = self.__get_storage_state()
+ storage_state = await self.__get_storage_state()
except Exception:
logger.exception("Error while reloading MSD state; switching to offline")
@@ -514,8 +514,8 @@ class Plugin(BaseMsd): # pylint: disable=too-many-instance-attributes
# =====
- def __get_storage_state(self) -> _StorageState:
- images = self.__storage.get_images()
+ async def __get_storage_state(self) -> _StorageState:
+ images = await self.__storage.get_images()
space = self.__storage.get_space(fatal=True)
assert space
return _StorageState(
diff --git a/kvmd/plugins/msd/otg/storage.py b/kvmd/plugins/msd/otg/storage.py
index 4cca878b..c6a3e34b 100644
--- a/kvmd/plugins/msd/otg/storage.py
+++ b/kvmd/plugins/msd/otg/storage.py
@@ -21,6 +21,7 @@
import os
+import operator
import dataclasses
from typing import Generator
@@ -31,6 +32,7 @@ import aiofiles.os
from ....logging import get_logger
+from .... import aiotools
from .... import aiohelpers
from .. import MsdError
@@ -130,26 +132,42 @@ class Storage:
self.__path = path
self.__remount_cmd = remount_cmd
- def get_watchable_paths(self) -> list[str]:
- paths: list[str] = []
- for (root_path, dirs, _) in os.walk(self.__path):
- dirs[:] = list(self.__filtered(dirs))
- paths.append(root_path)
- return paths
+ async def get_watchable_paths(self) -> list[str]:
+ return (await aiotools.run_async(self.__get_watchable_paths))
- def get_images(self) -> dict[str, Image]:
+ async def get_images(self) -> dict[str, Image]:
+ return (await aiotools.run_async(self.__get_images))
+
+ def __get_watchable_paths(self) -> list[str]:
+ return list(map(operator.itemgetter(0), self.__walk(with_files=False)))
+
+ def __get_images(self) -> dict[str, Image]:
images: dict[str, Image] = {}
- for (root_path, dirs, files) in os.walk(self.__path):
- dirs[:] = list(self.__filtered(dirs))
- for file in self.__filtered(files):
- name = os.path.relpath(os.path.join(root_path, file), self.__path)
+ for (_, files) in self.__walk(with_files=True):
+ for path in files:
+ name = os.path.relpath(path, self.__path)
images[name] = self.get_image_by_name(name)
return images
- def __filtered(self, items: list[str]) -> Generator[str, None, None]:
- for item in sorted(map(str.strip, items)):
- if not item.startswith(".") and item != "lost+found":
- yield item
+ def __walk(self, with_files: bool, root_path: (str | None)=None) -> Generator[tuple[str, list[str]], None, None]:
+ if root_path is None:
+ root_path = self.__path
+ files: list[str] = []
+ with os.scandir(root_path) as dir_iter:
+ for item in sorted(dir_iter, key=operator.attrgetter("name")):
+ if item.name.startswith(".") or item.name == "lost+found":
+ continue
+ try:
+ if item.is_dir(follow_symlinks=False):
+ item.stat() # Проверяем, не сдохла ли смонтированная NFS
+ yield from self.__walk(with_files, item.path)
+ elif with_files and item.is_file(follow_symlinks=False):
+ files.append(item.path)
+ except Exception:
+ pass
+ yield (root_path, files)
+
+ # =====
def get_image_by_name(self, name: str) -> Image:
assert name
@@ -170,6 +188,8 @@ class Storage:
assert path
return Image(name, path, (self if in_storage else None))
+ # =====
+
def get_space(self, fatal: bool) -> (StorageSpace | None):
try:
st = os.statvfs(self.__path)
diff --git a/kvmd/plugins/ugpio/otgconf.py b/kvmd/plugins/ugpio/otgconf.py
index e24c26b9..194475c3 100644
--- a/kvmd/plugins/ugpio/otgconf.py
+++ b/kvmd/plugins/ugpio/otgconf.py
@@ -82,8 +82,8 @@ class Plugin(BaseUserGpioDriver):
await asyncio.sleep(5)
with Inotify() as inotify:
- inotify.watch(os.path.dirname(self.__udc_path), InotifyMask.ALL_MODIFY_EVENTS)
- inotify.watch(self.__profile_path, InotifyMask.ALL_MODIFY_EVENTS)
+ await inotify.watch(os.path.dirname(self.__udc_path), InotifyMask.ALL_MODIFY_EVENTS)
+ await inotify.watch(self.__profile_path, InotifyMask.ALL_MODIFY_EVENTS)
self._notifier.notify()
while True:
need_restart = False