dycw-utilities 0.151.11__py3-none-any.whl → 0.152.0__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,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: dycw-utilities
3
- Version: 0.151.11
3
+ Version: 0.152.0
4
4
  Author-email: Derek Wan <d.wan@icloud.com>
5
5
  License-File: LICENSE
6
6
  Requires-Python: >=3.12
@@ -12,7 +12,7 @@ Provides-Extra: logging
12
12
  Requires-Dist: coloredlogs<15.1,>=15.0.1; extra == 'logging'
13
13
  Provides-Extra: test
14
14
  Requires-Dist: dycw-pytest-only<2.2,>=2.1.1; extra == 'test'
15
- Requires-Dist: hypothesis<6.137,>=6.136.1; extra == 'test'
15
+ Requires-Dist: hypothesis<6.137,>=6.136.6; extra == 'test'
16
16
  Requires-Dist: pudb<2025.2,>=2025.1; extra == 'test'
17
17
  Requires-Dist: pytest-asyncio<1.2,>=1.1.0; extra == 'test'
18
18
  Requires-Dist: pytest-cov<6.3,>=6.2.1; extra == 'test'
@@ -1,4 +1,4 @@
1
- utilities/__init__.py,sha256=QFtmxXQUOY461TnnnmD5VQHWJT3vLCqDXSEDvXwfrUM,61
1
+ utilities/__init__.py,sha256=yPinoOXl37WGTDQPnlJyr4YW7B9ojeL_vWBwenSrkaQ,60
2
2
  utilities/altair.py,sha256=92E2lCdyHY4Zb-vCw6rEJIsWdKipuu-Tu2ab1ufUfAk,9079
3
3
  utilities/asyncio.py,sha256=r75HVwm6QoCOz7bEe3_KO47G4_IIhNCYcPDrJTi4i_4,16777
4
4
  utilities/atomicwrites.py,sha256=tPo6r-Rypd9u99u66B9z86YBPpnLrlHtwox_8Z7T34Y,5790
@@ -11,7 +11,6 @@ utilities/contextvars.py,sha256=J8OhC7jqozAGYOCe2KUWysbPXNGe5JYz3HfaY_mIs08,883
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=G05UH-fqUbcRPjQ8arK6K0Ap2fRbzEm0SZahJKCqYfY,32643
14
- utilities/dnspython.py,sha256=HfFaxjA_25XhjsTM5Bg03GCFatKzL8dOV4E0c9oMsho,384
15
14
  utilities/enum.py,sha256=5l6pwZD1cjSlVW4ss-zBPspWvrbrYrdtJWcg6f5_J5w,5781
16
15
  utilities/errors.py,sha256=mFlDGSM0LI1jZ1pbqwLAH3ttLZ2JVIxyZLojw8tGVZU,1479
17
16
  utilities/eventkit.py,sha256=FRCZisJfY9hAS9GHV4ZO3ZHDs89XZod7Xf99WzBt7jQ,12636
@@ -27,7 +26,7 @@ utilities/hypothesis.py,sha256=m44niSfuzuhgn7IQ1UOwUGgiu68xz4a6LHxB0IE6fNE,40341
27
26
  utilities/importlib.py,sha256=mV1xT_O_zt_GnZZ36tl3xOmMaN_3jErDWY54fX39F6Y,429
28
27
  utilities/inflect.py,sha256=v7YkOWSu8NAmVghPcf4F3YBZQoJCS47_DLf9jbfWIs0,581
29
28
  utilities/ipython.py,sha256=V2oMYHvEKvlNBzxDXdLvKi48oUq2SclRg5xasjaXStw,763
30
- utilities/iterables.py,sha256=6PFrb2fx1As22iCnlcUv35heGVpnRpiSObJFUmO9GDc,43922
29
+ utilities/iterables.py,sha256=VGqnVk-Wq13WHZZQ8L6BCCwFzEOoKGRpd7xkiQaXbiY,43642
31
30
  utilities/json.py,sha256=-WcGtSsCr9Y42wHZzAMnfvU6ihAfVftylFfRUORaDFo,2102
32
31
  utilities/jupyter.py,sha256=ft5JA7fBxXKzP-L9W8f2-wbF0QeYc_2uLQNFDVk4Z-M,2917
33
32
  utilities/libcst.py,sha256=TKgKN4bNmtBNEE-TUfhTyd1BrTncfsl_7tTuhpesGYY,5585
@@ -47,7 +46,7 @@ utilities/pathlib.py,sha256=77wT9naY2Nnrbar8nJiIYd2r3MfabMQM9VguuuivrdQ,8481
47
46
  utilities/period.py,sha256=c4-N8N1GIUyomoHmSO7yPfYgK8tc__dxNSB79knlx2w,12471
48
47
  utilities/pickle.py,sha256=MBT2xZCsv0pH868IXLGKnlcqNx2IRVKYNpRcqiQQqxw,653
49
48
  utilities/platform.py,sha256=pTn7gw6N4T6LdKrf0virwarof_mze9WtoQlrGMzhGVI,2798
50
- utilities/polars.py,sha256=x4x3i08ldcDyKVEa8rcyAyacHKjok30K2v79yhoIn2Y,73243
49
+ utilities/polars.py,sha256=uxV4liIQoCO92aERDO-AMfa8Es_-xCKgwZoyTO3PazQ,74410
51
50
  utilities/polars_ols.py,sha256=Uc9V5kvlWZ5cU93lKZ-cfAKdVFFw81tqwLW9PxtUvMs,5618
52
51
  utilities/postgres.py,sha256=juXaOguCCX4oAw4y2JajRVIhhKKZarCyapNyyr4Tn4Q,12538
53
52
  utilities/pottery.py,sha256=u0uvyGgYyujxftEMlsv6ppYTKQoVVjHt5jnVxxYz9s4,6596
@@ -59,7 +58,7 @@ utilities/pytest.py,sha256=xR-xG5aTfU9U7Qcb8Lle8uhLP1V0zc8EqZFfU0F8zfg,7899
59
58
  utilities/pytest_regressions.py,sha256=ocjHTtfOeiGfQAKIei8pKNd61sxN9dawrJJ9gPt2wzA,4097
60
59
  utilities/random.py,sha256=YWYzWxQDeyJRiuHGnO1OxF6dDucpq7qc1tH_ealwCRg,4130
61
60
  utilities/re.py,sha256=S4h-DLL6ScMPqjboZ_uQ1BVTJajrqV06r_81D--_HCE,4573
62
- utilities/redis.py,sha256=lNVQWHN5j1Sf8A8CoUxG6kU27ZDStrjw3JXqGscqD6I,28404
61
+ utilities/redis.py,sha256=2fdveFbqL2pEAeyiVuN_Je8nSM_IZHeahPduMHhFRzY,28381
63
62
  utilities/reprlib.py,sha256=ssYTcBW-TeRh3fhCJv57sopTZHF5FrPyyUg9yp5XBlo,3953
64
63
  utilities/scipy.py,sha256=wZJM7fEgBAkLSYYvSmsg5ac-QuwAI0BGqHVetw1_Hb0,947
65
64
  utilities/sentinel.py,sha256=3jIwgpMekWgDAxPDA_hXMP2St43cPhciKN3LWiZ7kv0,1248
@@ -76,7 +75,7 @@ utilities/threading.py,sha256=GvBOp4CyhHfN90wGXZuA2VKe9fGzMaEa7oCl4f3nnPU,1009
76
75
  utilities/timer.py,sha256=oXfTii6ymu57niP0BDGZjFD55LEHi2a19kqZKiTgaFQ,2588
77
76
  utilities/traceback.py,sha256=zofhzIedpUHrzDNiRJDVzm_wuu_tlTQvVqK4quxVlgM,9151
78
77
  utilities/typed_settings.py,sha256=ZzzlMgPntnGv6kAn0mGp43nHJCnU99V_8152QmpH9tY,4476
79
- utilities/types.py,sha256=4bHcAF4OVMjnaY9SAj8_bKiHfj6YurD6Y3cipV-5yFE,18551
78
+ utilities/types.py,sha256=-DUlUxqkVGaSVU_uTbfMIXWUDDUXnPuIDwQIsPE_mY8,18244
80
79
  utilities/typing.py,sha256=Z-_XDaWyT_6wIo3qfNK-hvRlzxP2Jxa9PgXzm5rDYRA,13790
81
80
  utilities/tzdata.py,sha256=fgNVj66yUbCSI_-vrRVzSD3gtf-L_8IEJEPjP_Jel5Y,266
82
81
  utilities/tzlocal.py,sha256=KyCXEgCTjqGFx-389JdTuhMRUaT06U1RCMdWoED-qro,728
@@ -89,8 +88,8 @@ utilities/zoneinfo.py,sha256=FBMcUQ4662Aq8SsuCL1OAhDQiyANmVjtb-C30DRrWoE,1966
89
88
  utilities/pytest_plugins/__init__.py,sha256=U4S_2y3zgLZVfMenHRaJFBW8yqh2mUBuI291LGQVOJ8,35
90
89
  utilities/pytest_plugins/pytest_randomly.py,sha256=NXzCcGKbpgYouz5yehKb4jmxmi2SexKKpgF4M65bi10,414
91
90
  utilities/pytest_plugins/pytest_regressions.py,sha256=Iwhfv_OJH7UCPZCfoh7ugZ2Xjqjil-BBBsOb8sDwiGI,1471
92
- dycw_utilities-0.151.11.dist-info/METADATA,sha256=_R0ltbaJCdA-g4I1cTYINPyOhEa1S6ImCwHMAXdG0HI,1697
93
- dycw_utilities-0.151.11.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
94
- dycw_utilities-0.151.11.dist-info/entry_points.txt,sha256=BOD_SoDxwsfJYOLxhrSXhHP_T7iw-HXI9f2WVkzYxvQ,135
95
- dycw_utilities-0.151.11.dist-info/licenses/LICENSE,sha256=gppZp16M6nSVpBbUBrNL6JuYfvKwZiKgV7XoKKsHzqo,1066
96
- dycw_utilities-0.151.11.dist-info/RECORD,,
91
+ dycw_utilities-0.152.0.dist-info/METADATA,sha256=JV_YGZDAfL333osSK-mawT3eeu0RHzt2PNkd-09o0yM,1696
92
+ dycw_utilities-0.152.0.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
93
+ dycw_utilities-0.152.0.dist-info/entry_points.txt,sha256=BOD_SoDxwsfJYOLxhrSXhHP_T7iw-HXI9f2WVkzYxvQ,135
94
+ dycw_utilities-0.152.0.dist-info/licenses/LICENSE,sha256=gppZp16M6nSVpBbUBrNL6JuYfvKwZiKgV7XoKKsHzqo,1066
95
+ dycw_utilities-0.152.0.dist-info/RECORD,,
utilities/__init__.py CHANGED
@@ -1,3 +1,3 @@
1
1
  from __future__ import annotations
2
2
 
3
- __version__ = "0.151.11"
3
+ __version__ = "0.152.0"
utilities/iterables.py CHANGED
@@ -46,7 +46,7 @@ from utilities.types import SupportsAdd, SupportsLT
46
46
  if TYPE_CHECKING:
47
47
  from types import NoneType
48
48
 
49
- from utilities.types import MaybeIterable, MaybeIterableHashable, Sign, StrMapping
49
+ from utilities.types import MaybeIterable, Sign, StrMapping
50
50
 
51
51
 
52
52
  ##
@@ -66,16 +66,6 @@ def always_iterable[T](obj: MaybeIterable[T], /) -> Iterable[T]:
66
66
  ##
67
67
 
68
68
 
69
- def always_iterable_hashable[T](
70
- obj: MaybeIterable[T] | None, /
71
- ) -> MaybeIterableHashable[T] | None:
72
- """Ensure an object is always hashable."""
73
- return None if obj is None else tuple(always_iterable(obj))
74
-
75
-
76
- ##
77
-
78
-
79
69
  def apply_bijection[T, U](
80
70
  func: Callable[[T], U], iterable: Iterable[T], /
81
71
  ) -> Mapping[T, U]:
@@ -1489,7 +1479,6 @@ __all__ = [
1489
1479
  "ResolveIncludeAndExcludeError",
1490
1480
  "SortIterableError",
1491
1481
  "always_iterable",
1492
- "always_iterable_hashable",
1493
1482
  "apply_bijection",
1494
1483
  "apply_to_tuple",
1495
1484
  "apply_to_varargs",
utilities/polars.py CHANGED
@@ -7,7 +7,7 @@ from collections.abc import Set as AbstractSet
7
7
  from contextlib import suppress
8
8
  from dataclasses import asdict, dataclass
9
9
  from functools import partial, reduce
10
- from itertools import chain, product
10
+ from itertools import chain, pairwise, product
11
11
  from math import ceil, log
12
12
  from pathlib import Path
13
13
  from typing import TYPE_CHECKING, Any, Literal, assert_never, cast, overload, override
@@ -1471,6 +1471,37 @@ class _GetSeriesNumberOfDecimalsAllNullError(GetSeriesNumberOfDecimalsError):
1471
1471
  ##
1472
1472
 
1473
1473
 
1474
+ @overload
1475
+ def increasing_horizontal(*columns: ExprLike) -> Expr: ...
1476
+ @overload
1477
+ def increasing_horizontal(*columns: Series) -> Series: ...
1478
+ @overload
1479
+ def increasing_horizontal(*columns: IntoExprColumn) -> Expr | Series: ...
1480
+ def increasing_horizontal(*columns: IntoExprColumn) -> Expr | Series:
1481
+ """Check if a set of columns are increasing."""
1482
+ columns2 = ensure_expr_or_series_many(*columns)
1483
+ if len(columns2) == 0:
1484
+ return lit(value=True, dtype=Boolean)
1485
+ return all_horizontal(prev < curr for prev, curr in pairwise(columns2))
1486
+
1487
+
1488
+ @overload
1489
+ def decreasing_horizontal(*columns: ExprLike) -> Expr: ...
1490
+ @overload
1491
+ def decreasing_horizontal(*columns: Series) -> Series: ...
1492
+ @overload
1493
+ def decreasing_horizontal(*columns: IntoExprColumn) -> Expr | Series: ...
1494
+ def decreasing_horizontal(*columns: IntoExprColumn) -> Expr | Series:
1495
+ """Check if a set of columns are decreasing."""
1496
+ columns2 = ensure_expr_or_series_many(*columns)
1497
+ if len(columns2) == 0:
1498
+ return lit(value=True, dtype=Boolean)
1499
+ return all_horizontal(prev > curr for prev, curr in pairwise(columns2))
1500
+
1501
+
1502
+ ##
1503
+
1504
+
1474
1505
  def insert_after(df: DataFrame, column: str, value: IntoExprColumn, /) -> DataFrame:
1475
1506
  """Insert a series after an existing column; not in-place."""
1476
1507
  columns = df.columns
@@ -2472,6 +2503,7 @@ __all__ = [
2472
2503
  "cross",
2473
2504
  "dataclass_to_dataframe",
2474
2505
  "dataclass_to_schema",
2506
+ "decreasing_horizontal",
2475
2507
  "deserialize_dataframe",
2476
2508
  "drop_null_struct_series",
2477
2509
  "ensure_data_type",
@@ -2483,6 +2515,7 @@ __all__ = [
2483
2515
  "get_expr_name",
2484
2516
  "get_frequency_spectrum",
2485
2517
  "get_series_number_of_decimals",
2518
+ "increasing_horizontal",
2486
2519
  "insert_after",
2487
2520
  "insert_before",
2488
2521
  "insert_between",
utilities/redis.py CHANGED
@@ -35,7 +35,7 @@ if TYPE_CHECKING:
35
35
  from redis.typing import EncodableT
36
36
 
37
37
  from utilities.iterables import MaybeIterable
38
- from utilities.types import Delta, MaybeListStr, MaybeSequence, MaybeType, TypeLike
38
+ from utilities.types import Delta, MaybeSequence, MaybeType, TypeLike
39
39
 
40
40
 
41
41
  _PUBLISH_TIMEOUT: Delta = SECOND
@@ -583,7 +583,7 @@ class PublishError(Exception):
583
583
  async def publish_many[T](
584
584
  redis: Redis,
585
585
  channel: str,
586
- data: MaybeSequence[bytes | T] | MaybeListStr,
586
+ data: MaybeSequence[bytes | str | T],
587
587
  /,
588
588
  *,
589
589
  serializer: Callable[[T], EncodableT] | None = None,
utilities/types.py CHANGED
@@ -2,15 +2,7 @@ from __future__ import annotations
2
2
 
3
3
  import datetime as dt
4
4
  from asyncio import Event
5
- from collections.abc import (
6
- Callable,
7
- Collection,
8
- Coroutine,
9
- Hashable,
10
- Iterable,
11
- Mapping,
12
- Sequence,
13
- )
5
+ from collections.abc import Callable, Collection, Coroutine, Iterable, Mapping
14
6
  from enum import Enum
15
7
  from ipaddress import IPv4Address, IPv6Address
16
8
  from logging import Logger
@@ -123,20 +115,16 @@ IPv6AddressLike = MaybeStr[IPv6Address]
123
115
 
124
116
 
125
117
  # iterables
126
- type IterableHashable[T: Hashable] = tuple[T, ...] | frozenset[T]
127
118
  type MaybeCollection[T] = T | Collection[T]
128
- type MaybeIterableHashable[T: Hashable] = T | IterableHashable[T]
129
119
  type MaybeIterable[T] = T | Iterable[T]
130
- type MaybeList[T] = T | list[T]
131
- type MaybeSequence[T] = T | Sequence[T]
120
+ type MaybeSequence[T] = T | SequenceLT[T]
121
+ type SequenceLT[T] = list[T] | tuple[T, ...]
132
122
  # iterables - str
133
- type CollectionStr = (
134
- dict[str, Any] | frozenset[str] | list[str] | set[str] | tuple[str, ...]
135
- )
123
+ type SequenceStr = SequenceLT[str]
124
+ type CollectionStr = dict[str, Any] | frozenset[str] | set[str] | SequenceStr
125
+ # iterables - maybe str
136
126
  type MaybeCollectionStr = str | CollectionStr
137
- type MaybeListStr = MaybeList[str]
138
127
  type MaybeSequenceStr = str | SequenceStr
139
- type SequenceStr = list[str] | tuple[str, ...]
140
128
 
141
129
 
142
130
  # logging
@@ -288,7 +276,6 @@ __all__ = [
288
276
  "ExceptionTypeLike",
289
277
  "IPv4AddressLike",
290
278
  "IPv6AddressLike",
291
- "IterableHashable",
292
279
  "LogLevel",
293
280
  "LoggerOrName",
294
281
  "MathRoundMode",
@@ -303,9 +290,6 @@ __all__ = [
303
290
  "MaybeCollectionStr",
304
291
  "MaybeCoro",
305
292
  "MaybeIterable",
306
- "MaybeIterableHashable",
307
- "MaybeList",
308
- "MaybeListStr",
309
293
  "MaybeSequence",
310
294
  "MaybeSequenceStr",
311
295
  "MaybeStr",
utilities/dnspython.py DELETED
@@ -1,17 +0,0 @@
1
- from __future__ import annotations
2
-
3
- from ipaddress import IPv4Address
4
-
5
- from dns.resolver import resolve
6
-
7
- from utilities.functools import cache
8
-
9
-
10
- @cache
11
- def nslookup(address: str, /) -> list[IPv4Address]:
12
- """Look up a set of addresses."""
13
- ans = resolve(address) # skipif-ci-and-mac
14
- return [IPv4Address(str(rd)) for rd in ans] # skipif-ci-and-mac
15
-
16
-
17
- __all__ = ["nslookup"]