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.
- omlish/__about__.py +2 -2
- omlish/argparse.py +4 -4
- omlish/asyncs/anyio.py +62 -1
- omlish/asyncs/futures.py +6 -6
- omlish/c3.py +4 -4
- omlish/check.py +6 -6
- omlish/collections/__init__.py +98 -63
- omlish/collections/cache/descriptor.py +5 -5
- omlish/collections/cache/impl.py +4 -4
- omlish/collections/coerce.py +43 -43
- omlish/collections/frozen.py +3 -3
- omlish/collections/identity.py +1 -1
- omlish/collections/mappings.py +3 -3
- omlish/collections/ordered.py +1 -1
- omlish/collections/skiplist.py +6 -6
- omlish/collections/sorted.py +3 -3
- omlish/collections/treap.py +17 -17
- omlish/collections/treapmap.py +2 -2
- omlish/collections/unmodifiable.py +28 -27
- omlish/configs/flattening.py +1 -1
- omlish/configs/props.py +1 -1
- omlish/dataclasses/impl/__init__.py +2 -0
- omlish/dataclasses/impl/api.py +5 -13
- omlish/dataclasses/impl/fields.py +1 -1
- omlish/dataclasses/impl/init.py +1 -1
- omlish/dataclasses/impl/internals.py +15 -0
- omlish/dataclasses/impl/main.py +4 -4
- omlish/dataclasses/impl/metaclass.py +1 -1
- omlish/dataclasses/impl/metadata.py +1 -1
- omlish/dataclasses/impl/order.py +1 -1
- omlish/dataclasses/impl/params.py +4 -38
- omlish/dataclasses/impl/reflect.py +1 -7
- omlish/dataclasses/impl/repr.py +23 -5
- omlish/dataclasses/impl/simple.py +2 -2
- omlish/dataclasses/impl/slots.py +2 -2
- omlish/dataclasses/impl/utils.py +4 -4
- omlish/dispatch/dispatch.py +9 -8
- omlish/dispatch/methods.py +2 -2
- omlish/docker.py +8 -6
- omlish/dynamic.py +5 -5
- omlish/graphs/dot/items.py +1 -1
- omlish/graphs/trees.py +15 -21
- omlish/inject/elements.py +1 -1
- omlish/inject/exceptions.py +1 -1
- omlish/inject/impl/injector.py +1 -1
- omlish/inject/impl/inspect.py +1 -1
- omlish/inject/injector.py +1 -1
- omlish/inject/providers.py +2 -2
- omlish/iterators.py +43 -2
- omlish/lang/__init__.py +167 -112
- omlish/lang/cached.py +13 -5
- omlish/lang/classes/__init__.py +35 -24
- omlish/lang/classes/abstract.py +1 -1
- omlish/lang/classes/simple.py +1 -1
- omlish/lang/clsdct.py +1 -1
- omlish/lang/contextmanagers.py +23 -15
- omlish/lang/datetimes.py +1 -1
- omlish/lang/descriptors.py +35 -2
- omlish/lang/exceptions.py +2 -0
- omlish/lang/functions.py +43 -13
- omlish/lang/imports.py +8 -8
- omlish/lang/iterables.py +1 -1
- omlish/lang/maybes.py +1 -1
- omlish/lang/objects.py +2 -2
- omlish/lang/timeouts.py +53 -0
- omlish/lang/typing.py +2 -2
- omlish/libc.py +6 -6
- omlish/marshal/base.py +6 -6
- omlish/marshal/dataclasses.py +2 -2
- omlish/marshal/enums.py +2 -2
- omlish/marshal/factories.py +10 -10
- omlish/marshal/iterables.py +2 -2
- omlish/marshal/mappings.py +2 -2
- omlish/marshal/optionals.py +4 -4
- omlish/marshal/polymorphism.py +4 -4
- omlish/marshal/standard.py +6 -6
- omlish/marshal/utils.py +1 -1
- omlish/os.py +13 -4
- omlish/procfs.py +336 -0
- omlish/reflect.py +2 -12
- omlish/replserver/console.py +9 -9
- omlish/replserver/server.py +4 -4
- omlish/sql/__init__.py +0 -0
- omlish/sql/_abcs.py +65 -0
- omlish/sql/dbs.py +90 -0
- omlish/stats.py +3 -3
- omlish/testing/pydevd.py +4 -6
- omlish/testing/pytest/inject/__init__.py +7 -0
- omlish/testing/pytest/inject/harness.py +23 -1
- omlish/testing/pytest/plugins/__init__.py +1 -1
- omlish/testing/pytest/plugins/pydevd.py +12 -0
- omlish/testing/pytest/plugins/switches.py +2 -2
- omlish/testing/testing.py +5 -5
- omlish/text/parts.py +3 -3
- omlish-0.0.0.dev2.dist-info/METADATA +31 -0
- omlish-0.0.0.dev2.dist-info/RECORD +193 -0
- {omlish-0.0.0.dev1.dist-info → omlish-0.0.0.dev2.dist-info}/WHEEL +1 -1
- omlish/testing/pytest/plugins/pycharm.py +0 -54
- omlish-0.0.0.dev1.dist-info/METADATA +0 -17
- omlish-0.0.0.dev1.dist-info/RECORD +0 -187
- {omlish-0.0.0.dev1.dist-info → omlish-0.0.0.dev2.dist-info}/LICENSE +0 -0
- {omlish-0.0.0.dev1.dist-info → omlish-0.0.0.dev2.dist-info}/top_level.txt +0 -0
omlish/__about__.py
CHANGED
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:
|
|
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:
|
|
96
|
-
parent:
|
|
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.
|
|
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:
|
|
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) ->
|
|
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:
|
|
48
|
-
tick_interval_s:
|
|
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:
|
|
82
|
-
tick_interval_s:
|
|
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:
|
|
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.
|
|
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.
|
|
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[...,
|
|
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:
|
|
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) ->
|
|
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:
|
|
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:
|
|
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
|
omlish/collections/__init__.py
CHANGED
|
@@ -1,63 +1,98 @@
|
|
|
1
|
-
from .coerce import
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
from .frozen import
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
from .
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
from .
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
from .
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
from .
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
from .
|
|
62
|
-
|
|
63
|
-
|
|
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:
|
|
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
|
omlish/collections/cache/impl.py
CHANGED
|
@@ -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.
|
|
86
|
-
value: ta.
|
|
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[[
|
|
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.
|
|
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)
|