Report.aggregate: param data -> iter get_data

This commit is contained in:
Jörn-Michael Miehe 2023-08-31 15:31:25 +00:00
parent 8b532829a5
commit 99d879ba85
5 changed files with 46 additions and 48 deletions

View file

@ -1,5 +1,5 @@
from dataclasses import dataclass, field from dataclasses import dataclass
from typing import Self from typing import Callable, Iterator, Self
from ..settings import MetricSettings from ..settings import MetricSettings
@ -17,35 +17,39 @@ class ReportData:
) )
def report(self, settings: MetricSettings) -> "Report": def report(self, settings: MetricSettings) -> "Report":
result = settings.report.format( return Report(
name=self.name, result=settings.report.format(
value=self.value, name=self.name,
value=self.value,
),
failed=(
self.value > settings.threshold and not settings.inverted
or self.value < settings.threshold and settings.inverted
),
) )
return Report(result, failed=(
self.value > settings.threshold and not settings.inverted
or self.value < settings.threshold and settings.inverted
))
@dataclass(slots=True, kw_only=True, frozen=True)
@dataclass(slots=True, frozen=True)
class Report: class Report:
result: str result: str
failed: bool = field(default=False, kw_only=True) failed: bool = False
@classmethod @classmethod
def aggregate( def aggregate(
cls, *, cls, *,
settings: MetricSettings, settings: MetricSettings,
data: list[ReportData], get_data: Callable[[], Iterator[ReportData]],
) -> Self: ) -> Self | None:
if not settings.enabled:
return None
reports = [ reports = [
data.report(settings) data.report(settings)
for data in data for data in get_data()
] ]
return cls( return cls(
settings.report_outer.format( result=settings.report_outer.format(
name=settings.name, name=settings.name,
inner=", ".join( inner=", ".join(
report.result report.result

View file

@ -1,20 +1,20 @@
from typing import Iterator
import psutil import psutil
from ..settings import SETTINGS from ..settings import SETTINGS
from ._report import Report, ReportData from ._report import Report, ReportData
def _hwdata() -> ReportData: def _hwdata() -> Iterator[ReportData]:
return ReportData( yield ReportData(
name=SETTINGS.cpu.name, name=SETTINGS.cpu.name,
value=psutil.cpu_percent(interval=1), value=psutil.cpu_percent(interval=1),
) )
def cpu() -> Report | None: def cpu() -> Report | None:
if not SETTINGS.cpu.enabled: return Report.aggregate(
return None settings=SETTINGS.cpu,
get_data=_hwdata,
data = _hwdata() )
return data.report(SETTINGS.cpu)

View file

@ -1,10 +1,11 @@
import os import os
from typing import Iterator
from ..settings import SETTINGS from ..settings import SETTINGS
from ._report import Report, ReportData from ._report import Report, ReportData
def _hwdata() -> list[ReportData]: def _hwdata() -> Iterator[ReportData]:
def get_path_statvfs(path: os.PathLike) -> dict[str, float]: def get_path_statvfs(path: os.PathLike) -> dict[str, float]:
sv = os.statvfs(path) sv = os.statvfs(path)
return { return {
@ -12,7 +13,7 @@ def _hwdata() -> list[ReportData]:
"total": sv.f_blocks, "total": sv.f_blocks,
} }
return sorted([ yield from sorted([
ReportData.from_free_total( ReportData.from_free_total(
name=str(path), name=str(path),
**get_path_statvfs(path), **get_path_statvfs(path),
@ -21,12 +22,7 @@ def _hwdata() -> list[ReportData]:
def disk() -> Report | None: def disk() -> Report | None:
if not SETTINGS.disk.enabled:
return None
data = _hwdata()
return Report.aggregate( return Report.aggregate(
settings=SETTINGS.disk, settings=SETTINGS.disk,
data=data, get_data=_hwdata,
) )

View file

@ -1,43 +1,41 @@
from typing import Iterator
import psutil import psutil
from ..settings import SETTINGS from ..settings import SETTINGS
from ._report import Report, ReportData from ._report import Report, ReportData
def _hwdata() -> list[ReportData]: def _hwdata() -> Iterator[ReportData]:
vmem = psutil.virtual_memory() vmem = psutil.virtual_memory()
swap = psutil.swap_memory() swap = psutil.swap_memory()
if SETTINGS.memory.swap == "exclude": if SETTINGS.memory.swap == "exclude":
return [ReportData( yield ReportData(
name=SETTINGS.memory.name_ram, name=SETTINGS.memory.name_ram,
value=vmem.percent, value=vmem.percent,
)] )
elif SETTINGS.memory.swap == "combine": elif SETTINGS.memory.swap == "combine":
return [ReportData.from_free_total( yield ReportData.from_free_total(
name=SETTINGS.memory.name, name=SETTINGS.memory.name,
free=vmem.available + swap.free, free=vmem.available + swap.free,
total=vmem.total + swap.total, total=vmem.total + swap.total,
)] )
else: # SETTINGS.memory.swap == "include" else: # SETTINGS.memory.swap == "include"
return [ReportData( yield ReportData(
name=SETTINGS.memory.name_ram, name=SETTINGS.memory.name_ram,
value=vmem.percent, value=vmem.percent,
), ReportData( )
yield ReportData(
name=SETTINGS.memory.name_swap, name=SETTINGS.memory.name_swap,
value=swap.percent, value=swap.percent,
)] )
def memory() -> Report | None: def memory() -> Report | None:
if not SETTINGS.memory.enabled:
return None
data = _hwdata()
return Report.aggregate( return Report.aggregate(
settings=SETTINGS.memory, settings=SETTINGS.memory,
data=data, get_data=_hwdata,
) )

View file

@ -22,7 +22,7 @@ class MetricSettings(BaseModel):
report: str = "{name}: {value:.2f}%" report: str = "{name}: {value:.2f}%"
# per-metric format string for reporting # per-metric format string for reporting
report_outer: str = "{name}: [{inner}]" report_outer: str = "{inner}"
# include only `count` many items (None: include all) # include only `count` many items (None: include all)
count: int | None = None count: int | None = None
@ -36,7 +36,6 @@ class CpuMS(MetricSettings):
class MemoryMS(MetricSettings): class MemoryMS(MetricSettings):
name: str = "Memory" name: str = "Memory"
threshold: float = 90 threshold: float = 90
report_outer: str = "{inner}"
# how to handle swap space # how to handle swap space
# exclude: swap space is not reported # exclude: swap space is not reported
@ -52,6 +51,7 @@ class MemoryMS(MetricSettings):
class DiskMS(MetricSettings): class DiskMS(MetricSettings):
name: str = "Disk Used" name: str = "Disk Used"
threshold: float = 85 threshold: float = 85
report_outer: str = "{name}: [{inner}]"
count: int = 1 count: int = 1
# paths to check for disk space # paths to check for disk space