dycw-utilities 0.125.20__py3-none-any.whl → 0.125.22__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,12 +1,12 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: dycw-utilities
3
- Version: 0.125.20
3
+ Version: 0.125.22
4
4
  Author-email: Derek Wan <d.wan@icloud.com>
5
5
  License-File: LICENSE
6
6
  Requires-Python: >=3.12
7
7
  Requires-Dist: typing-extensions<4.14,>=4.13.1
8
8
  Provides-Extra: test
9
- Requires-Dist: hypothesis<6.132,>=6.131.30; extra == 'test'
9
+ Requires-Dist: hypothesis<6.132,>=6.131.32; extra == 'test'
10
10
  Requires-Dist: pytest-asyncio<1.1,>=1.0.0; extra == 'test'
11
11
  Requires-Dist: pytest-cov<6.2,>=6.1.1; extra == 'test'
12
12
  Requires-Dist: pytest-instafail<0.6,>=0.5.0; extra == 'test'
@@ -78,7 +78,7 @@ Provides-Extra: zzz-test-hypothesis
78
78
  Requires-Dist: aiosqlite<0.22,>=0.21.0; extra == 'zzz-test-hypothesis'
79
79
  Requires-Dist: asyncpg<0.31,>=0.30.0; extra == 'zzz-test-hypothesis'
80
80
  Requires-Dist: greenlet<3.3,>=3.2.0; extra == 'zzz-test-hypothesis'
81
- Requires-Dist: hypothesis<6.132,>=6.131.30; extra == 'zzz-test-hypothesis'
81
+ Requires-Dist: hypothesis<6.132,>=6.131.32; extra == 'zzz-test-hypothesis'
82
82
  Requires-Dist: luigi<3.7,>=3.6.0; extra == 'zzz-test-hypothesis'
83
83
  Requires-Dist: numpy<2.3,>=2.2.6; extra == 'zzz-test-hypothesis'
84
84
  Requires-Dist: pathvalidate<3.3,>=3.2.3; extra == 'zzz-test-hypothesis'
@@ -1,4 +1,4 @@
1
- utilities/__init__.py,sha256=7jjcydn2HpkHk2ldvyVCqxPJFAZwBb7SiotXzuwh50k,61
1
+ utilities/__init__.py,sha256=mOuoQC76dPEs9bAglMk52el5SSD3hguWO9Zm_CqFpl4,61
2
2
  utilities/altair.py,sha256=Gpja-flOo-Db0PIPJLJsgzAlXWoKUjPU1qY-DQ829ek,9156
3
3
  utilities/asyncio.py,sha256=pQ5GRcNyBqmDyeAeSarYugrjSNxmX3E0G9pye-kkPt4,51085
4
4
  utilities/atomicwrites.py,sha256=geFjn9Pwn-tTrtoGjDDxWli9NqbYfy3gGL6ZBctiqSo,5393
@@ -6,14 +6,14 @@ utilities/atools.py,sha256=IYMuFSFGSKyuQmqD6v5IUtDlz8PPw0Sr87Cub_gRU3M,1168
6
6
  utilities/cachetools.py,sha256=C1zqOg7BYz0IfQFK8e3qaDDgEZxDpo47F15RTfJM37Q,2910
7
7
  utilities/click.py,sha256=SRVkUkWxyO_7AaYvvDl0hNCzdpneX04lf1O97i9MfPw,14311
8
8
  utilities/concurrent.py,sha256=s2scTEd2AhXVTW4hpASU2qxV_DiVLALfms55cCQzCvM,2886
9
- utilities/contextlib.py,sha256=OOIIEa5lXKGzFAnauaul40nlQnQko6Na4ryiMJcHkIg,478
9
+ utilities/contextlib.py,sha256=lpaLJBy3X0UGLWjM98jkQZZq8so4fRmoK-Bheq0uOW4,1027
10
10
  utilities/contextvars.py,sha256=RsSGGrbQqqZ67rOydnM7WWIsM2lIE31UHJLejnHJPWY,505
11
11
  utilities/cryptography.py,sha256=_CiK_K6c_-uQuUhsUNjNjTL-nqxAh4_1zTfS11Xe120,972
12
12
  utilities/cvxpy.py,sha256=Rv1-fD-XYerosCavRF8Pohop2DBkU3AlFaGTfD8AEAA,13776
13
13
  utilities/dataclasses.py,sha256=iiC1wpGXWhaocIikzwBt8bbLWyImoUlOlcDZJGejaIg,33011
14
14
  utilities/datetime.py,sha256=uYoaOi_C1YtNXGfTN9xlTrW62Re2b1_4Skuv14_MeYQ,38985
15
15
  utilities/enum.py,sha256=HoRwVCWzsnH0vpO9ZEcAAIZLMv0Sn2vJxxA4sYMQgDs,5793
16
- utilities/errors.py,sha256=C89Xm-DHUjfqJKROo1pW-RGat8zDvTzZuy6zBNMkj_s,837
16
+ utilities/errors.py,sha256=wg6Jrgz5T_KbmG_PXF6pbDFSFso9CG93RGz1Om6Q0Lk,848
17
17
  utilities/eventkit.py,sha256=6M5Xu1SzN-juk9PqBHwy5dS-ta7T0qA6SMpDsakOJ0E,13039
18
18
  utilities/fastapi.py,sha256=LG1-Q8RDi7wsyVN6v74qptPYX8WGXPkFOQFniMvtzjc,2439
19
19
  utilities/fpdf2.py,sha256=y1NGXR5chWqLXWpewGV3hlRGMr_5yV1lVRkPBhPEgJI,1843
@@ -50,6 +50,7 @@ utilities/platform.py,sha256=48IOKx1IC6ZJXWG-b56ZQptITcNFhWRjELW72o2dGTA,2398
50
50
  utilities/polars.py,sha256=QlmUpYTqHNkcLnWOQh1TW22W2QyLzvifCvBcbsqhpdE,63272
51
51
  utilities/polars_ols.py,sha256=Uc9V5kvlWZ5cU93lKZ-cfAKdVFFw81tqwLW9PxtUvMs,5618
52
52
  utilities/pqdm.py,sha256=foRytQybmOQ05pjt5LF7ANyzrIa--4ScDE3T2wd31a4,3118
53
+ utilities/psutil.py,sha256=0dXHKW7t9J3uzY3YtHGiRiZVRWr0uozccioZRCVFQfk,3761
53
54
  utilities/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
54
55
  utilities/pydantic.py,sha256=f6qtR5mO2YMuyvNmbaEj5YeD9eGA4YYfb7Bjzh9jUs0,1845
55
56
  utilities/pyinstrument.py,sha256=OJFDh4o1CWIa4aYPYURdQjgap_nvP45KUsCEe94rQHY,829
@@ -89,7 +90,7 @@ utilities/warnings.py,sha256=un1LvHv70PU-LLv8RxPVmugTzDJkkGXRMZTE2-fTQHw,1771
89
90
  utilities/whenever.py,sha256=jS31ZAY5OMxFxLja_Yo5Fidi87Pd-GoVZ7Vi_teqVDA,16743
90
91
  utilities/zipfile.py,sha256=24lQc9ATcJxHXBPc_tBDiJk48pWyRrlxO2fIsFxU0A8,699
91
92
  utilities/zoneinfo.py,sha256=-5j7IQ9nb7gR43rdgA7ms05im-XuqhAk9EJnQBXxCoQ,1874
92
- dycw_utilities-0.125.20.dist-info/METADATA,sha256=xMxMTQI22apLwASoFNF6uBV9K2oGR4cBmxYYzv6wzKs,12852
93
- dycw_utilities-0.125.20.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
94
- dycw_utilities-0.125.20.dist-info/licenses/LICENSE,sha256=gppZp16M6nSVpBbUBrNL6JuYfvKwZiKgV7XoKKsHzqo,1066
95
- dycw_utilities-0.125.20.dist-info/RECORD,,
93
+ dycw_utilities-0.125.22.dist-info/METADATA,sha256=1cHkVeipeZn1O9uboTuGQ1v8cOEjkg3GIBqUnmmaIow,12852
94
+ dycw_utilities-0.125.22.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
95
+ dycw_utilities-0.125.22.dist-info/licenses/LICENSE,sha256=gppZp16M6nSVpBbUBrNL6JuYfvKwZiKgV7XoKKsHzqo,1066
96
+ dycw_utilities-0.125.22.dist-info/RECORD,,
utilities/__init__.py CHANGED
@@ -1,3 +1,3 @@
1
1
  from __future__ import annotations
2
2
 
3
- __version__ = "0.125.20"
3
+ __version__ = "0.125.22"
utilities/contextlib.py CHANGED
@@ -1,12 +1,17 @@
1
1
  from __future__ import annotations
2
2
 
3
+ import re
4
+ from contextlib import contextmanager
3
5
  from typing import TYPE_CHECKING
4
6
 
5
7
  if TYPE_CHECKING:
8
+ from collections.abc import Iterator
6
9
  from types import TracebackType
7
10
 
8
11
 
9
12
  class NoOpContextManager:
13
+ """Context-manager for no-op."""
14
+
10
15
  def __enter__(self) -> None:
11
16
  return None
12
17
 
@@ -20,4 +25,20 @@ class NoOpContextManager:
20
25
  return False
21
26
 
22
27
 
23
- __all__ = ["NoOpContextManager"]
28
+ ##
29
+
30
+
31
+ _SUPER_OBJECT_HAS_NO_ATTRIBUTE = re.compile(r"'super' object has no attribute '\w+'")
32
+
33
+
34
+ @contextmanager
35
+ def suppress_super_object_attribute_error() -> Iterator[None]:
36
+ """Suppress the super() attribute error, for mix-ins."""
37
+ try:
38
+ yield
39
+ except AttributeError as error:
40
+ if not _SUPER_OBJECT_HAS_NO_ATTRIBUTE.search(error.args[0]):
41
+ raise
42
+
43
+
44
+ __all__ = ["NoOpContextManager", "suppress_super_object_attribute_error"]
utilities/errors.py CHANGED
@@ -15,6 +15,7 @@ class ImpossibleCaseError(Exception):
15
15
  def __str__(self) -> str:
16
16
  desc = ", ".join(self.case)
17
17
  return f"Case must be possible: {desc}."
18
+ ##
18
19
 
19
20
 
20
21
  ##
utilities/psutil.py ADDED
@@ -0,0 +1,115 @@
1
+ from __future__ import annotations
2
+
3
+ from dataclasses import dataclass, field
4
+ from json import dumps
5
+ from logging import getLogger
6
+ from math import isclose, nan
7
+ from pathlib import Path
8
+ from typing import TYPE_CHECKING, Self, override
9
+
10
+ from psutil import swap_memory, virtual_memory
11
+
12
+ from utilities.asyncio import Looper
13
+ from utilities.contextlib import suppress_super_object_attribute_error
14
+ from utilities.datetime import SECOND, get_now
15
+
16
+ if TYPE_CHECKING:
17
+ import datetime as dt
18
+ from logging import Logger
19
+
20
+ from utilities.types import Duration, PathLike
21
+
22
+
23
+ @dataclass(kw_only=True)
24
+ class MemoryMonitorService(Looper[None]):
25
+ """Service to monitor memory usage."""
26
+
27
+ # base
28
+ freq: Duration = field(default=10 * SECOND, repr=False)
29
+ backoff: Duration = field(default=10 * SECOND, repr=False)
30
+ # self
31
+ console: str | None = field(default=None, repr=False)
32
+ path: PathLike = "memory.txt"
33
+ _console: Logger | None = field(init=False, repr=False)
34
+ _max_age: int | None = field(default=None, init=False, repr=False)
35
+ _path: Path = field(init=False, repr=False)
36
+
37
+ @override
38
+ def __post_init__(self) -> None:
39
+ super().__post_init__()
40
+ if self.console is not None:
41
+ self._console = getLogger(self.console)
42
+ self._path = Path(self.path)
43
+
44
+ @override
45
+ async def core(self) -> None:
46
+ await super().core()
47
+ memory = MemoryUsage.new()
48
+ mapping = {
49
+ "datetime": memory.datetime.strftime("%Y-%m-%d %H:%M:%S"),
50
+ "virtual used (mb)": memory.virtual_used_mb,
51
+ "virtual total (mb)": memory.virtual_total_mb,
52
+ "virtual (%)": memory.virtual_pct,
53
+ "swap used (mb)": memory.swap_used_mb,
54
+ "swap total (mb)": memory.swap_total_mb,
55
+ "swap (%)": memory.swap_pct,
56
+ }
57
+ ser = dumps(mapping)
58
+ with self._path.open(mode="a") as fh:
59
+ _ = fh.write(f"{ser}\n")
60
+ if self._console is not None:
61
+ self._console.info("%s", mapping)
62
+
63
+
64
+ ##
65
+
66
+
67
+ @dataclass(kw_only=True)
68
+ class MemoryUsage:
69
+ """A memory usage."""
70
+
71
+ datetime: dt.datetime = field(default_factory=get_now)
72
+ virtual_used: int = field(repr=False)
73
+ virtual_used_mb: int = field(init=False)
74
+ virtual_total: int = field(repr=False)
75
+ virtual_total_mb: int = field(init=False)
76
+ virtual_pct: float = field(init=False)
77
+ swap_used: int = field(repr=False)
78
+ swap_used_mb: int = field(init=False)
79
+ swap_total: int = field(repr=False)
80
+ swap_total_mb: int = field(init=False)
81
+ swap_pct: float = field(init=False)
82
+
83
+ def __post_init__(self) -> None:
84
+ with suppress_super_object_attribute_error():
85
+ super().__post_init__() # pyright: ignore[reportAttributeAccessIssue]
86
+ self.virtual_used_mb = self._to_mb(self.virtual_used)
87
+ self.virtual_total_mb = self._to_mb(self.virtual_total)
88
+ self.virtual_pct = (
89
+ nan
90
+ if isclose(self.virtual_total, 0.0)
91
+ else self.virtual_used / self.virtual_total
92
+ )
93
+ self.swap_used_mb = self._to_mb(self.swap_used)
94
+ self.swap_total_mb = self._to_mb(self.swap_total)
95
+ self.swap_pct = (
96
+ nan if isclose(self.swap_total, 0.0) else self.swap_used / self.swap_total
97
+ )
98
+
99
+ @classmethod
100
+ def new(cls) -> Self:
101
+ virtual = virtual_memory()
102
+ virtual_total = virtual.total
103
+ swap = swap_memory()
104
+ return cls(
105
+ virtual_used=virtual_total - virtual.available,
106
+ virtual_total=virtual_total,
107
+ swap_used=swap.used,
108
+ swap_total=swap.total,
109
+ )
110
+
111
+ def _to_mb(self, bytes_: int) -> int:
112
+ return round(bytes_ / (1024**2))
113
+
114
+
115
+ __all__ = ["MemoryMonitorService", "MemoryUsage"]