omlish 0.0.0.dev1__py3-none-any.whl → 0.0.0.dev2__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.

Potentially problematic release.


This version of omlish might be problematic. Click here for more details.

Files changed (102) hide show
  1. omlish/__about__.py +2 -2
  2. omlish/argparse.py +4 -4
  3. omlish/asyncs/anyio.py +62 -1
  4. omlish/asyncs/futures.py +6 -6
  5. omlish/c3.py +4 -4
  6. omlish/check.py +6 -6
  7. omlish/collections/__init__.py +98 -63
  8. omlish/collections/cache/descriptor.py +5 -5
  9. omlish/collections/cache/impl.py +4 -4
  10. omlish/collections/coerce.py +43 -43
  11. omlish/collections/frozen.py +3 -3
  12. omlish/collections/identity.py +1 -1
  13. omlish/collections/mappings.py +3 -3
  14. omlish/collections/ordered.py +1 -1
  15. omlish/collections/skiplist.py +6 -6
  16. omlish/collections/sorted.py +3 -3
  17. omlish/collections/treap.py +17 -17
  18. omlish/collections/treapmap.py +2 -2
  19. omlish/collections/unmodifiable.py +28 -27
  20. omlish/configs/flattening.py +1 -1
  21. omlish/configs/props.py +1 -1
  22. omlish/dataclasses/impl/__init__.py +2 -0
  23. omlish/dataclasses/impl/api.py +5 -13
  24. omlish/dataclasses/impl/fields.py +1 -1
  25. omlish/dataclasses/impl/init.py +1 -1
  26. omlish/dataclasses/impl/internals.py +15 -0
  27. omlish/dataclasses/impl/main.py +4 -4
  28. omlish/dataclasses/impl/metaclass.py +1 -1
  29. omlish/dataclasses/impl/metadata.py +1 -1
  30. omlish/dataclasses/impl/order.py +1 -1
  31. omlish/dataclasses/impl/params.py +4 -38
  32. omlish/dataclasses/impl/reflect.py +1 -7
  33. omlish/dataclasses/impl/repr.py +23 -5
  34. omlish/dataclasses/impl/simple.py +2 -2
  35. omlish/dataclasses/impl/slots.py +2 -2
  36. omlish/dataclasses/impl/utils.py +4 -4
  37. omlish/dispatch/dispatch.py +9 -8
  38. omlish/dispatch/methods.py +2 -2
  39. omlish/docker.py +8 -6
  40. omlish/dynamic.py +5 -5
  41. omlish/graphs/dot/items.py +1 -1
  42. omlish/graphs/trees.py +15 -21
  43. omlish/inject/elements.py +1 -1
  44. omlish/inject/exceptions.py +1 -1
  45. omlish/inject/impl/injector.py +1 -1
  46. omlish/inject/impl/inspect.py +1 -1
  47. omlish/inject/injector.py +1 -1
  48. omlish/inject/providers.py +2 -2
  49. omlish/iterators.py +43 -2
  50. omlish/lang/__init__.py +167 -112
  51. omlish/lang/cached.py +13 -5
  52. omlish/lang/classes/__init__.py +35 -24
  53. omlish/lang/classes/abstract.py +1 -1
  54. omlish/lang/classes/simple.py +1 -1
  55. omlish/lang/clsdct.py +1 -1
  56. omlish/lang/contextmanagers.py +23 -15
  57. omlish/lang/datetimes.py +1 -1
  58. omlish/lang/descriptors.py +35 -2
  59. omlish/lang/exceptions.py +2 -0
  60. omlish/lang/functions.py +43 -13
  61. omlish/lang/imports.py +8 -8
  62. omlish/lang/iterables.py +1 -1
  63. omlish/lang/maybes.py +1 -1
  64. omlish/lang/objects.py +2 -2
  65. omlish/lang/timeouts.py +53 -0
  66. omlish/lang/typing.py +2 -2
  67. omlish/libc.py +6 -6
  68. omlish/marshal/base.py +6 -6
  69. omlish/marshal/dataclasses.py +2 -2
  70. omlish/marshal/enums.py +2 -2
  71. omlish/marshal/factories.py +10 -10
  72. omlish/marshal/iterables.py +2 -2
  73. omlish/marshal/mappings.py +2 -2
  74. omlish/marshal/optionals.py +4 -4
  75. omlish/marshal/polymorphism.py +4 -4
  76. omlish/marshal/standard.py +6 -6
  77. omlish/marshal/utils.py +1 -1
  78. omlish/os.py +13 -4
  79. omlish/procfs.py +336 -0
  80. omlish/reflect.py +2 -12
  81. omlish/replserver/console.py +9 -9
  82. omlish/replserver/server.py +4 -4
  83. omlish/sql/__init__.py +0 -0
  84. omlish/sql/_abcs.py +65 -0
  85. omlish/sql/dbs.py +90 -0
  86. omlish/stats.py +3 -3
  87. omlish/testing/pydevd.py +4 -6
  88. omlish/testing/pytest/inject/__init__.py +7 -0
  89. omlish/testing/pytest/inject/harness.py +23 -1
  90. omlish/testing/pytest/plugins/__init__.py +1 -1
  91. omlish/testing/pytest/plugins/pydevd.py +12 -0
  92. omlish/testing/pytest/plugins/switches.py +2 -2
  93. omlish/testing/testing.py +5 -5
  94. omlish/text/parts.py +3 -3
  95. omlish-0.0.0.dev2.dist-info/METADATA +31 -0
  96. omlish-0.0.0.dev2.dist-info/RECORD +193 -0
  97. {omlish-0.0.0.dev1.dist-info → omlish-0.0.0.dev2.dist-info}/WHEEL +1 -1
  98. omlish/testing/pytest/plugins/pycharm.py +0 -54
  99. omlish-0.0.0.dev1.dist-info/METADATA +0 -17
  100. omlish-0.0.0.dev1.dist-info/RECORD +0 -187
  101. {omlish-0.0.0.dev1.dist-info → omlish-0.0.0.dev2.dist-info}/LICENSE +0 -0
  102. {omlish-0.0.0.dev1.dist-info → omlish-0.0.0.dev2.dist-info}/top_level.txt +0 -0
omlish/__about__.py CHANGED
@@ -2,6 +2,6 @@ __name__ = 'omlish'
2
2
  __author__ = 'wrmsr'
3
3
  __url__ = 'https://github.com/wrmsr/omlish'
4
4
  __license__ = 'BSD-3-Clause'
5
- __requires_python__ = '>=3.11'
5
+ __requires_python__ = '>=3.12'
6
6
 
7
- __version__ = '0.0.0.dev1'
7
+ __version__ = '0.0.0.dev2'
omlish/argparse.py CHANGED
@@ -48,7 +48,7 @@ ArgumentParser = argparse.ArgumentParser
48
48
  class Arg:
49
49
  args: ta.Sequence[ta.Any]
50
50
  kwargs: ta.Mapping[str, ta.Any]
51
- dest: ta.Optional[str] = None
51
+ dest: str | None = None
52
52
 
53
53
  def __get__(self, instance, owner=None):
54
54
  if instance is None:
@@ -92,8 +92,8 @@ class Command:
92
92
 
93
93
  def command(
94
94
  *args: Arg,
95
- name: ta.Optional[str] = None,
96
- parent: ta.Optional[Command] = None,
95
+ name: str | None = None,
96
+ parent: Command | None = None,
97
97
  ) -> ta.Any: # ta.Callable[[CommandFn], Command]: # FIXME
98
98
  for arg in args:
99
99
  check.isinstance(arg, Arg)
@@ -192,7 +192,7 @@ class _CliMeta(type):
192
192
 
193
193
  class Cli(metaclass=_CliMeta):
194
194
 
195
- def __init__(self, argv: ta.Optional[ta.Sequence[str]] = None) -> None:
195
+ def __init__(self, argv: ta.Sequence[str] | None = None) -> None:
196
196
  super().__init__()
197
197
 
198
198
  self._argv = argv if argv is not None else sys.argv[1:]
omlish/asyncs/anyio.py CHANGED
@@ -7,10 +7,15 @@ lookit:
7
7
  - https://github.com/M-o-a-T/asyncscope
8
8
  - https://github.com/M-o-a-T/aevent
9
9
  - https://github.com/florimondmanca/aiometer
10
+ - https://github.com/sanitizers/octomachinery/blob/b36c3d3d49da813ac46e361424132955a4e99ac8/octomachinery/utils/asynctools.py
10
11
  """ # noqa
11
12
  import typing as ta
12
13
 
13
- import anyio
14
+ import anyio.streams.memory
15
+ import anyio.streams.stapled
16
+
17
+ from .. import check
18
+ from .. import lang
14
19
 
15
20
 
16
21
  T = ta.TypeVar('T')
@@ -21,3 +26,59 @@ async def anyio_eof_to_empty(fn: ta.Callable[..., ta.Awaitable[T]], *args: ta.An
21
26
  return await fn(*args, **kwargs)
22
27
  except anyio.EndOfStream:
23
28
  return b''
29
+
30
+
31
+ def split_memory_object_streams(
32
+ *args: anyio.create_memory_object_stream[T],
33
+ ) -> tuple[
34
+ anyio.streams.memory.MemoryObjectSendStream[T],
35
+ anyio.streams.memory.MemoryObjectReceiveStream[T],
36
+ ]:
37
+ [tup] = args
38
+ return tup
39
+
40
+
41
+ # FIXME: https://github.com/python/mypy/issues/15238
42
+ # def create_memory_object_stream[T](max_buffer_size: float = 0) -> tuple[
43
+ # anyio.streams.memory.MemoryObjectSendStream[T],
44
+ # anyio.streams.memory.MemoryObjectReceiveStream[T],
45
+ # ]:
46
+ # return anyio.create_memory_object_stream[T](max_buffer_size)
47
+
48
+
49
+ def staple_memory_object_stream(
50
+ *args: anyio.create_memory_object_stream[T],
51
+ ) -> anyio.streams.stapled.StapledObjectStream[T]:
52
+ send, receive = args
53
+ return anyio.streams.stapled.StapledObjectStream(
54
+ check.isinstance(send, anyio.streams.memory.MemoryObjectSendStream), # type: ignore
55
+ check.isinstance(receive, anyio.streams.memory.MemoryObjectReceiveStream), # type: ignore
56
+ )
57
+
58
+
59
+ # FIXME: https://github.com/python/mypy/issues/15238
60
+ # def staple_memory_object_stream2[T](max_buffer_size: float = 0) -> anyio.streams.stapled.StapledObjectStream[T]:
61
+ # send, receive = anyio.create_memory_object_stream[T](max_buffer_size)
62
+ # return anyio.streams.stapled.StapledObjectStream(
63
+ # check.isinstance(send, anyio.streams.memory.MemoryObjectSendStream), # type: ignore
64
+ # check.isinstance(receive, anyio.streams.memory.MemoryObjectReceiveStream), # type: ignore
65
+ # )
66
+
67
+
68
+ async def gather(*fns: ta.Callable[..., ta.Awaitable[T]], take_first: bool = False) -> list[lang.Maybe[T]]:
69
+ results: list[lang.Maybe[T]] = [lang.empty()] * len(fns)
70
+
71
+ async def inner(fn, i):
72
+ results[i] = lang.just(await fn())
73
+ if take_first:
74
+ tg.cancel_scope.cancel()
75
+
76
+ async with anyio.create_task_group() as tg:
77
+ for i, fn in enumerate(fns):
78
+ tg.start_soon(inner, fn, i)
79
+
80
+ return results
81
+
82
+
83
+ async def first(*fns: ta.Callable[..., ta.Awaitable[T]], **kwargs: ta.Any) -> list[lang.Maybe[T]]:
84
+ return await gather(*fns, take_first=True, **kwargs)
omlish/asyncs/futures.py CHANGED
@@ -12,7 +12,7 @@ T = ta.TypeVar('T')
12
12
 
13
13
  class FutureException(Exception, ta.Generic[T]):
14
14
 
15
- def __init__(self, future: cf.Future, target: ta.Optional[T] = None) -> None:
15
+ def __init__(self, future: cf.Future, target: T | None = None) -> None:
16
16
  super().__init__()
17
17
 
18
18
  self._future = future
@@ -23,7 +23,7 @@ class FutureException(Exception, ta.Generic[T]):
23
23
  return self._future
24
24
 
25
25
  @property
26
- def target(self) -> ta.Optional[T]:
26
+ def target(self) -> T | None:
27
27
  return self._target
28
28
 
29
29
  def __repr__(self) -> str:
@@ -44,8 +44,8 @@ class FutureTimeoutException(Exception):
44
44
  def wait_futures(
45
45
  futures: ta.Sequence[cf.Future],
46
46
  *,
47
- timeout_s: ta.Union[int, float] = 60,
48
- tick_interval_s: ta.Union[int, float] = 0.5,
47
+ timeout_s: float = 60,
48
+ tick_interval_s: float = .5,
49
49
  tick_fn: ta.Callable[..., bool] = lambda: True,
50
50
  raise_exceptions: bool = False,
51
51
  cancel_on_exception: bool = False,
@@ -78,8 +78,8 @@ def wait_dependent_futures(
78
78
  executor: cf.Executor,
79
79
  dependency_sets_by_fn: ta.Mapping[ta.Callable, ta.AbstractSet[ta.Callable]],
80
80
  *,
81
- timeout_s: ta.Union[int, float] = 60,
82
- tick_interval_s: ta.Union[int, float] = 0.5,
81
+ timeout_s: float = 60,
82
+ tick_interval_s: float = .5,
83
83
  tick_fn: ta.Callable[..., bool] = lambda: True,
84
84
  ) -> ta.Mapping[ta.Callable, cf.Future]:
85
85
  for fn, deps in dependency_sets_by_fn.items():
omlish/c3.py CHANGED
@@ -49,7 +49,7 @@ def merge(seqs: ta.MutableSequence[list[T]]) -> list[T]:
49
49
  """
50
50
 
51
51
  result: list[T] = []
52
- candidate: ta.Optional[T] = None
52
+ candidate: T | None = None
53
53
  while True:
54
54
  seqs = [s for s in seqs if s] # purge empty sequences
55
55
  if not seqs:
@@ -73,7 +73,7 @@ def merge(seqs: ta.MutableSequence[list[T]]) -> list[T]:
73
73
 
74
74
  def mro(
75
75
  cls: T,
76
- abcs: ta.Optional[ta.Sequence[T]] = None,
76
+ abcs: ta.Sequence[T] | None = None,
77
77
  *,
78
78
  get_bases: ta.Callable[[T], ta.Sequence[T]] = operator.attrgetter('__bases__'),
79
79
  is_subclass: ta.Callable[[T, T], bool] = issubclass, # type: ignore
@@ -115,7 +115,7 @@ def mro(
115
115
  return merge(
116
116
  [[cls]] +
117
117
  explicit_c3_mros + abstract_c3_mros + other_c3_mros +
118
- [explicit_bases] + [abstract_bases] + [other_bases]
118
+ [explicit_bases] + [abstract_bases] + [other_bases],
119
119
  )
120
120
 
121
121
 
@@ -123,7 +123,7 @@ def compose_mro(
123
123
  cls: T,
124
124
  types: ta.Iterable[T],
125
125
  *,
126
- get_mro: ta.Callable[[T], ta.Optional[ta.Sequence[T]]] = operator.attrgetter('__mro__'),
126
+ get_mro: ta.Callable[[T], ta.Sequence[T] | None] = operator.attrgetter('__mro__'),
127
127
  get_bases: ta.Callable[[T], ta.Sequence[T]] = operator.attrgetter('__bases__'),
128
128
  is_subclass: ta.Callable[[T, T], bool] = issubclass, # type: ignore
129
129
  get_subclasses: ta.Callable[[T], ta.Iterable[T]] = operator.methodcaller('__subclasses__'),
omlish/check.py CHANGED
@@ -9,7 +9,7 @@ import typing as ta
9
9
  T = ta.TypeVar('T')
10
10
  SizedT = ta.TypeVar('SizedT', bound=ta.Sized)
11
11
 
12
- Message = ta.Union[str, ta.Callable[..., ta.Optional[str]], None]
12
+ Message = ta.Union[str, ta.Callable[..., str | None], None]
13
13
 
14
14
  _NONE_TYPE = type(None)
15
15
 
@@ -33,7 +33,7 @@ def _raise(
33
33
  default_message: str,
34
34
  message: Message,
35
35
  *args: ta.Any,
36
- **kwargs: ta.Any
36
+ **kwargs: ta.Any,
37
37
  ) -> ta.NoReturn:
38
38
  if _callable(message):
39
39
  message = ta.cast(ta.Callable, message)(*args, **kwargs)
@@ -64,7 +64,7 @@ def isinstance(v: ta.Any, spec: ta.Union[type[T], tuple], msg: Message = None) -
64
64
  return v
65
65
 
66
66
 
67
- def of_isinstance(spec: ta.Union[type[T], tuple], msg: Message = None) -> ta.Callable[[ta.Any], T]:
67
+ def of_isinstance(spec: type[T] | tuple, msg: Message = None) -> ta.Callable[[ta.Any], T]:
68
68
  def inner(v):
69
69
  return isinstance(v, _unpack_isinstance_spec(spec), msg)
70
70
 
@@ -155,7 +155,7 @@ def single(obj: ta.Iterable[T], message: Message = None) -> T:
155
155
  return value
156
156
 
157
157
 
158
- def optional_single(obj: ta.Iterable[T], message: Message = None) -> ta.Optional[T]:
158
+ def optional_single(obj: ta.Iterable[T], message: Message = None) -> T | None:
159
159
  it = iter(obj)
160
160
  try:
161
161
  value = next(it)
@@ -176,7 +176,7 @@ def none(v: ta.Any, msg: Message = None) -> None:
176
176
  _raise(ValueError, 'Must be None', msg, v)
177
177
 
178
178
 
179
- def not_none(v: ta.Optional[T], msg: Message = None) -> T:
179
+ def not_none(v: T | None, msg: Message = None) -> T:
180
180
  if v is None:
181
181
  _raise(ValueError, 'Must not be None', msg, v)
182
182
  return v
@@ -212,7 +212,7 @@ def callable(v: T, msg: Message = None) -> T:
212
212
  return v # type: ignore
213
213
 
214
214
 
215
- def non_empty_str(v: ta.Optional[str], msg: Message = None) -> str:
215
+ def non_empty_str(v: str | None, msg: Message = None) -> str:
216
216
  if not _isinstance(v, str) or not v:
217
217
  _raise(ValueError, 'Must be non-empty str', msg, v)
218
218
  return v
@@ -1,63 +1,98 @@
1
- from .coerce import abs_set # noqa
2
- from .coerce import abs_set_of # noqa
3
- from .coerce import abs_set_of_or_none # noqa
4
- from .coerce import abs_set_or_none # noqa
5
- from .coerce import frozenset_ # noqa
6
- from .coerce import frozenset_of # noqa
7
- from .coerce import frozenset_of_or_none # noqa
8
- from .coerce import frozenset_or_none # noqa
9
- from .coerce import map # noqa
10
- from .coerce import map_of # noqa
11
- from .coerce import map_of_or_none # noqa
12
- from .coerce import map_or_none # noqa
13
- from .coerce import opt_abs_set # noqa
14
- from .coerce import opt_abs_set_of # noqa
15
- from .coerce import opt_frozenset # noqa
16
- from .coerce import opt_frozenset_of # noqa
17
- from .coerce import opt_map # noqa
18
- from .coerce import opt_map_of # noqa
19
- from .coerce import opt_seq # noqa
20
- from .coerce import opt_seq_of # noqa
21
- from .coerce import seq # noqa
22
- from .coerce import seq_of # noqa
23
- from .coerce import seq_of_or_none # noqa
24
- from .coerce import seq_or_none # noqa
25
- from .frozen import Frozen # noqa
26
- from .frozen import FrozenDict # noqa
27
- from .frozen import FrozenList # noqa
28
- from .frozen import frozendict # noqa
29
- from .frozen import frozenlist # noqa
30
- from .identity import IdentityKeyDict # noqa
31
- from .identity import IdentitySet # noqa
32
- from .identity import IdentityWrapper # noqa
33
- from .indexed import IndexedSeq # noqa
34
- from .indexed import IndexedSetSeq # noqa
35
- from .mappings import MissingDict # noqa
36
- from .mappings import TypeMap # noqa
37
- from .mappings import TypeMultiMap # noqa
38
- from .mappings import guarded_map_update # noqa
39
- from .mappings import multikey_dict # noqa
40
- from .mappings import yield_dict_init # noqa
41
- from .ordered import OrderedFrozenSet # noqa
42
- from .ordered import OrderedSet # noqa
43
- from .persistent import PersistentMap # noqa
44
- from .skiplist import SkipList # noqa
45
- from .skiplist import SkipListDict # noqa
46
- from .sorted import SortedCollection # noqa
47
- from .sorted import SortedListDict # noqa
48
- from .sorted import SortedMapping # noqa
49
- from .sorted import SortedMutableMapping # noqa
50
- from .treapmap import new_treap_map # noqa
51
- from .unmodifiable import Unmodifiable # noqa
52
- from .unmodifiable import UnmodifiableMapping # noqa
53
- from .unmodifiable import UnmodifiableSequence # noqa
54
- from .unmodifiable import UnmodifiableSet # noqa
55
- from .utils import all_equal # noqa
56
- from .utils import all_not_equal # noqa
57
- from .utils import indexes # noqa
58
- from .utils import key_cmp # noqa
59
- from .utils import mut_toposort # noqa
60
- from .utils import partition # noqa
61
- from .utils import toposort # noqa
62
- from .utils import unique # noqa
63
- from .utils import unique_dict # noqa
1
+ from .coerce import ( # noqa
2
+ abs_set,
3
+ abs_set_of,
4
+ abs_set_of_or_none,
5
+ abs_set_or_none,
6
+ frozenset_,
7
+ frozenset_of,
8
+ frozenset_of_or_none,
9
+ frozenset_or_none,
10
+ map,
11
+ map_of,
12
+ map_of_or_none,
13
+ map_or_none,
14
+ opt_abs_set,
15
+ opt_abs_set_of,
16
+ opt_frozenset,
17
+ opt_frozenset_of,
18
+ opt_map,
19
+ opt_map_of,
20
+ opt_seq,
21
+ opt_seq_of,
22
+ seq,
23
+ seq_of,
24
+ seq_of_or_none,
25
+ seq_or_none,
26
+ )
27
+
28
+ from .frozen import ( # noqa
29
+ Frozen,
30
+ FrozenDict,
31
+ FrozenList,
32
+ frozendict,
33
+ frozenlist,
34
+ )
35
+
36
+ from .identity import ( # noqa
37
+ IdentityKeyDict,
38
+ IdentitySet,
39
+ IdentityWrapper,
40
+ )
41
+
42
+ from .indexed import ( # noqa
43
+ IndexedSeq,
44
+ IndexedSetSeq,
45
+ )
46
+
47
+ from .mappings import ( # noqa
48
+ MissingDict,
49
+ TypeMap,
50
+ TypeMultiMap,
51
+ guarded_map_update,
52
+ multikey_dict,
53
+ yield_dict_init,
54
+ )
55
+
56
+ from .ordered import ( # noqa
57
+ OrderedFrozenSet,
58
+ OrderedSet,
59
+ )
60
+
61
+ from .persistent import ( # noqa
62
+ PersistentMap,
63
+ )
64
+
65
+ from .skiplist import ( # noqa
66
+ SkipList,
67
+ SkipListDict,
68
+ )
69
+
70
+ from .sorted import ( # noqa
71
+ SortedCollection,
72
+ SortedListDict,
73
+ SortedMapping,
74
+ SortedMutableMapping,
75
+ )
76
+
77
+ from .treapmap import ( # noqa
78
+ new_treap_map,
79
+ )
80
+
81
+ from .unmodifiable import ( # noqa
82
+ Unmodifiable,
83
+ UnmodifiableMapping,
84
+ UnmodifiableSequence,
85
+ UnmodifiableSet,
86
+ )
87
+
88
+ from .utils import ( # noqa
89
+ all_equal,
90
+ all_not_equal,
91
+ indexes,
92
+ key_cmp,
93
+ mut_toposort,
94
+ partition,
95
+ toposort,
96
+ unique,
97
+ unique_dict,
98
+ )
@@ -27,7 +27,7 @@ class _HashedSeq(list):
27
27
  def __init__(
28
28
  self,
29
29
  tup: ta.Tuple,
30
- hasher: ta.Callable[[ta.Any], int] = hash
30
+ hasher: ta.Callable[[ta.Any], int] = hash,
31
31
  ) -> None:
32
32
  super().__init__()
33
33
 
@@ -46,7 +46,7 @@ def _make_key(
46
46
  fasttypes=frozenset([int, str, frozenset, type(None)]),
47
47
  tuple=tuple,
48
48
  type=type,
49
- len=len
49
+ len=len,
50
50
  ) -> ta.Any:
51
51
  key = args
52
52
  if kwargs:
@@ -81,7 +81,7 @@ class _CacheDescriptor:
81
81
  fn: ta.Callable,
82
82
  scope: Scope,
83
83
  typed: bool,
84
- **kwargs: ta.Any
84
+ **kwargs: ta.Any,
85
85
  ) -> None:
86
86
  super().__init__()
87
87
 
@@ -175,9 +175,9 @@ class _CacheDescriptor:
175
175
 
176
176
 
177
177
  def cache(
178
- scope: ta.Union[Scope, str] = Scope.INSTANCE,
178
+ scope: Scope | str = Scope.INSTANCE,
179
179
  typed: bool = False,
180
- **kwargs
180
+ **kwargs,
181
181
  ) -> CC:
182
182
  if not isinstance(scope, Scope):
183
183
  scope = getattr(Scope, scope.upper()) # noqa
@@ -82,8 +82,8 @@ class CacheImpl(Cache[K, V]):
82
82
  lru_next: 'CacheImpl.Link'
83
83
  lfu_prev: 'CacheImpl.Link'
84
84
  lfu_next: 'CacheImpl.Link'
85
- key: ta.Union[ta.Any, weakref.ref]
86
- value: ta.Union[ta.Any, weakref.ref]
85
+ key: ta.Any | weakref.ref
86
+ value: ta.Any | weakref.ref
87
87
  weight: float
88
88
  written: float
89
89
  accessed: float
@@ -120,7 +120,7 @@ class CacheImpl(Cache[K, V]):
120
120
  identity_keys: bool = False,
121
121
  expire_after_access: float | None = None,
122
122
  expire_after_write: float | None = None,
123
- removal_listener: ta.Callable[[ta.Union[K, weakref.ref], ta.Union[V, weakref.ref]], None] | None = None,
123
+ removal_listener: ta.Callable[[K | weakref.ref, V | weakref.ref], None] | None = None,
124
124
  clock: ta.Callable[[], float] | None = None,
125
125
  weak_keys: bool = False,
126
126
  weak_values: bool = False,
@@ -165,7 +165,7 @@ class CacheImpl(Cache[K, V]):
165
165
  if self._track_frequency:
166
166
  self._root.lfu_next = self._root.lfu_prev = self._root
167
167
 
168
- weak_dead: ta.Optional[ta.Deque[CacheImpl.Link]]
168
+ weak_dead: ta.Deque[CacheImpl.Link] | None
169
169
  if weak_keys or weak_values:
170
170
  weak_dead = collections.deque()
171
171
  weak_dead_ref = weakref.ref(weak_dead)