mirror of
https://codeberg.org/privacy1st/de-p1st-monitor
synced 2025-04-02 12:45:59 +02:00
165 lines
5.5 KiB
Python
165 lines
5.5 KiB
Python
from pathlib import Path
|
|
|
|
import psutil
|
|
from de.p1st.monitor import datetime_util
|
|
from de.p1st.monitor.exec_capture import execute_capture
|
|
|
|
from de.p1st.monitor.logger import Logger
|
|
from de.p1st.monitor.logger_ex import LoggerArgEx, LoggerReadEx
|
|
from de.p1st.monitor.warn import WarnLevel, WarnMessage
|
|
|
|
|
|
class NotMounted(Exception):
|
|
pass
|
|
|
|
|
|
class FilesystemLogger(Logger):
|
|
def __init__(self, uuid: str = None,
|
|
mountpoint: Path = None,
|
|
unmounted_ok: bool = False,
|
|
warn_if_above: float = 1.0,
|
|
warn_threshold: int = 1,
|
|
warn_data_range: int = 1,
|
|
):
|
|
|
|
# The space between disk is at `self.warn_if_above` and disk is full at `1.0`.
|
|
buffer = 1 - warn_if_above
|
|
critical_if_above = warn_if_above + 0.5 * buffer
|
|
super().__init__(warn_threshold,
|
|
warn_data_range,
|
|
warn_if_above,
|
|
critical_if_above,
|
|
)
|
|
|
|
if uuid is None and mountpoint is None:
|
|
raise LoggerArgEx('uuid or mountpoint required')
|
|
|
|
self.uuid = uuid
|
|
self.mountpoint = mountpoint
|
|
self.unmounted_ok = unmounted_ok
|
|
self.warn_if_above = warn_if_above
|
|
|
|
self.mounted = True
|
|
|
|
#
|
|
#
|
|
#
|
|
|
|
# If uuid and mountpoint are both specified,
|
|
# raise warning if unexpected uuid is mounted at mountpoint.
|
|
if self.mountpoint is not None and self.uuid is not None:
|
|
try:
|
|
actual_uuid = self.get_uuid(self.mountpoint)
|
|
self.mounted = True
|
|
if self.uuid != actual_uuid:
|
|
raise LoggerReadEx(f'Expected {self.uuid} at {self.mountpoint} but got {actual_uuid}')
|
|
except NotMounted as e:
|
|
if self.unmounted_ok:
|
|
self.mounted = False
|
|
else:
|
|
raise LoggerArgEx(getattr(e, 'message', e))
|
|
|
|
# Try to get UUID (if only mountpoint given)
|
|
if self.uuid is None:
|
|
try:
|
|
self.uuid = self.get_uuid(self.mountpoint)
|
|
self.mounted = True
|
|
except NotMounted as e:
|
|
if self.unmounted_ok:
|
|
self.mounted = False
|
|
else:
|
|
raise LoggerArgEx(getattr(e, 'message', e))
|
|
|
|
# Try to get mountpoint (if only uuid given)
|
|
if self.mountpoint is None:
|
|
try:
|
|
self.mountpoint = self.get_mountpoint(self.uuid)
|
|
self.mounted = True
|
|
except NotMounted as e:
|
|
if self.unmounted_ok:
|
|
self.mounted = False
|
|
else:
|
|
raise LoggerReadEx(getattr(e, 'message', e))
|
|
|
|
def check_data(self, data: list[any]) -> WarnMessage:
|
|
if not self.mounted:
|
|
return WarnMessage(WarnLevel.NONE)
|
|
|
|
disk_usage = data[1]
|
|
message = f'Disk usage of {self.uuid} ist at {disk_usage}'
|
|
|
|
if disk_usage > self.critical_if_above:
|
|
return WarnMessage(WarnLevel.HIGH, message)
|
|
if disk_usage > self.warn_if_above:
|
|
return WarnMessage(WarnLevel.NORMAL, message)
|
|
return WarnMessage(WarnLevel.NONE)
|
|
|
|
def read_data(self) -> list[any] | None:
|
|
if not self.mounted:
|
|
return None
|
|
|
|
disk_usage: float = self.get_disk_usage(self.mountpoint)
|
|
return [
|
|
datetime_util.now(),
|
|
disk_usage,
|
|
]
|
|
|
|
def data_schema(self) -> list[str]:
|
|
return ['datetime#Date', 'float#Disk usage']
|
|
|
|
def get_log_file(self) -> Path:
|
|
# The mountpoint of a filesystem might change overtime.
|
|
# Thus, we use self.uuid to identify a filesystem.
|
|
return self.get_log_dir() / f'filesystem_{self.uuid}.csv'
|
|
|
|
#
|
|
# HELPERS
|
|
#
|
|
|
|
@classmethod
|
|
def get_disk_usage(cls, mountpoint: Path) -> float:
|
|
"""
|
|
:returns: used space / total space
|
|
"""
|
|
return psutil.disk_usage(str(mountpoint)).percent / 100.0
|
|
|
|
@classmethod
|
|
def get_mountpoint(cls, uuid: str) -> Path:
|
|
"""
|
|
Throws an error if the corresponding partition is not mounted.
|
|
"""
|
|
|
|
partition_list: list[psutil._common.sdiskpart] = psutil.disk_partitions(all=False)
|
|
partitions: dict[Path, psutil._common.sdiskpart] = {Path(partition.device).resolve(): partition for partition in
|
|
partition_list}
|
|
|
|
partition_path = cls.get_partition_path(uuid)
|
|
if partition_path not in partitions:
|
|
raise NotMounted(
|
|
f'Partition {partition_path} is probably not mounted '
|
|
f'as it is not in psutil partition list: {partitions}')
|
|
|
|
partition = partitions[partition_path]
|
|
return Path(partition.mountpoint)
|
|
|
|
@classmethod
|
|
def get_uuid(cls, mountpoint: Path) -> str:
|
|
# Returns the UUID of the device mounted at `/`.
|
|
# Fails if there is no disk mounted at `/`.
|
|
#
|
|
# findmnt / -o UUID -n
|
|
|
|
returncode, stdout, stderr = execute_capture(['findmnt', str(mountpoint), '-o', 'UUID', '-n'])
|
|
if returncode != 0:
|
|
raise NotMounted(
|
|
f'No partition mounted at {mountpoint}. Stderr of findmnt: {stderr}')
|
|
|
|
return stdout.strip()
|
|
|
|
@classmethod
|
|
def get_partition_path(cls, uuid: str) -> Path:
|
|
"""
|
|
:return: Partition path, e.g. /dev/sda1
|
|
"""
|
|
return Path(f'/dev/disk/by-uuid/{uuid}').resolve()
|