omlish 0.0.0.dev134__py3-none-any.whl → 0.0.0.dev137__py3-none-any.whl
Sign up to get free protection for your applications and to get access to all the features.
- omlish/.manifests.json +0 -12
- omlish/__about__.py +2 -2
- omlish/cached.py +2 -2
- omlish/collections/mappings.py +1 -1
- omlish/diag/_pycharm/runhack.py +3 -0
- omlish/formats/json/stream/lex.py +1 -1
- omlish/formats/json/stream/parse.py +1 -1
- omlish/{genmachine.py → funcs/genmachine.py} +4 -2
- omlish/{matchfns.py → funcs/match.py} +1 -1
- omlish/{fnpairs.py → funcs/pairs.py} +3 -3
- omlish/http/sessions.py +1 -1
- omlish/io/compress/__init__.py +0 -0
- omlish/io/compress/abc.py +104 -0
- omlish/io/compress/adapters.py +147 -0
- omlish/io/compress/bz2.py +41 -0
- omlish/io/compress/gzip.py +301 -0
- omlish/io/compress/lzma.py +31 -0
- omlish/io/compress/types.py +29 -0
- omlish/io/generators.py +50 -0
- omlish/lang/__init__.py +8 -1
- omlish/lang/generators.py +182 -0
- omlish/lang/iterables.py +28 -51
- omlish/lang/maybes.py +4 -4
- omlish/lite/fdio/corohttp.py +5 -1
- omlish/lite/marshal.py +9 -6
- omlish/marshal/base.py +1 -1
- omlish/marshal/factories.py +1 -1
- omlish/marshal/forbidden.py +1 -1
- omlish/marshal/iterables.py +1 -1
- omlish/marshal/mappings.py +1 -1
- omlish/marshal/maybes.py +1 -1
- omlish/marshal/standard.py +1 -1
- omlish/marshal/unions.py +1 -1
- omlish/secrets/pwhash.py +1 -1
- omlish/secrets/subprocesses.py +3 -1
- omlish/specs/jsonrpc/marshal.py +1 -1
- omlish/specs/openapi/marshal.py +1 -1
- {omlish-0.0.0.dev134.dist-info → omlish-0.0.0.dev137.dist-info}/METADATA +1 -1
- {omlish-0.0.0.dev134.dist-info → omlish-0.0.0.dev137.dist-info}/RECORD +49 -47
- omlish/formats/json/cli/__main__.py +0 -11
- omlish/formats/json/cli/cli.py +0 -298
- omlish/formats/json/cli/formats.py +0 -71
- omlish/formats/json/cli/io.py +0 -74
- omlish/formats/json/cli/parsing.py +0 -82
- omlish/formats/json/cli/processing.py +0 -48
- omlish/formats/json/cli/rendering.py +0 -92
- /omlish/collections/{_abc.py → abc.py} +0 -0
- /omlish/{formats/json/cli → funcs}/__init__.py +0 -0
- /omlish/{fnpipes.py → funcs/pipes.py} +0 -0
- /omlish/io/{_abc.py → abc.py} +0 -0
- /omlish/logs/{_abc.py → abc.py} +0 -0
- /omlish/sql/{_abc.py → abc.py} +0 -0
- {omlish-0.0.0.dev134.dist-info → omlish-0.0.0.dev137.dist-info}/LICENSE +0 -0
- {omlish-0.0.0.dev134.dist-info → omlish-0.0.0.dev137.dist-info}/WHEEL +0 -0
- {omlish-0.0.0.dev134.dist-info → omlish-0.0.0.dev137.dist-info}/entry_points.txt +0 -0
- {omlish-0.0.0.dev134.dist-info → omlish-0.0.0.dev137.dist-info}/top_level.txt +0 -0
omlish/.manifests.json
CHANGED
@@ -23,18 +23,6 @@
|
|
23
23
|
}
|
24
24
|
}
|
25
25
|
},
|
26
|
-
{
|
27
|
-
"module": ".formats.json.cli.__main__",
|
28
|
-
"attr": "_CLI_MODULE",
|
29
|
-
"file": "omlish/formats/json/cli/__main__.py",
|
30
|
-
"line": 1,
|
31
|
-
"value": {
|
32
|
-
"$omdev.cli.types.CliModule": {
|
33
|
-
"cmd_name": "json",
|
34
|
-
"mod_name": "omlish.formats.json.cli.__main__"
|
35
|
-
}
|
36
|
-
}
|
37
|
-
},
|
38
26
|
{
|
39
27
|
"module": ".secrets.pwgen",
|
40
28
|
"attr": "_CLI_MODULE",
|
omlish/__about__.py
CHANGED
omlish/cached.py
CHANGED
@@ -10,9 +10,9 @@ builtins and thus not distinguish it from a normal property.
|
|
10
10
|
|
11
11
|
"""
|
12
12
|
from .lang.cached import _CachedProperty # noqa
|
13
|
-
from .lang.cached import cached_function
|
13
|
+
from .lang.cached import cached_function as _cached_function
|
14
14
|
|
15
|
-
function =
|
15
|
+
function = _cached_function
|
16
16
|
|
17
17
|
property = property # noqa
|
18
18
|
|
omlish/collections/mappings.py
CHANGED
@@ -86,7 +86,7 @@ class DynamicTypeMap(ta.Generic[V]):
|
|
86
86
|
self._items = list(items)
|
87
87
|
self._weak = bool(weak)
|
88
88
|
|
89
|
-
self._cache: ta.MutableMapping[type, ta.Any] = weakref.WeakKeyDictionary()
|
89
|
+
self._cache: ta.MutableMapping[type, ta.Any] = weakref.WeakKeyDictionary() if weak else {}
|
90
90
|
|
91
91
|
@property
|
92
92
|
def items(self) -> ta.Sequence[V]:
|
omlish/diag/_pycharm/runhack.py
CHANGED
@@ -3,7 +3,10 @@
|
|
3
3
|
.venv/bin/python $(curl -LsSf https://raw.githubusercontent.com/wrmsr/omlish/master/omlish/diag/_pycharm/runhack.py -o $(mktemp) && echo "$_") install
|
4
4
|
|
5
5
|
==
|
6
|
+
TODO:
|
7
|
+
- check for existing files - can't run regular dep entrypoints now
|
6
8
|
|
9
|
+
==
|
7
10
|
See:
|
8
11
|
- https://github.com/JetBrains/intellij-community/blob/6400f70dde6f743e39a257a5a78cc51b644c835e/python/helpers/pycharm/_jb_pytest_runner.py
|
9
12
|
- https://github.com/JetBrains/intellij-community/blob/5a4e584aa59767f2e7cf4bd377adfaaf7503984b/python/helpers/pycharm/_jb_runner_tools.py
|
@@ -12,7 +12,7 @@ import typing as ta
|
|
12
12
|
I = ta.TypeVar('I')
|
13
13
|
O = ta.TypeVar('O')
|
14
14
|
|
15
|
-
# MachineGen: ta.TypeAlias = ta.Generator[ta.Iterable[O] | None, I, ta.Optional[MachineGen[I, O]]]
|
15
|
+
# MachineGen: ta.TypeAlias = ta.Generator[ta.Iterable[O] | None, I | None, ta.Optional[MachineGen[I, O]]]
|
16
16
|
MachineGen: ta.TypeAlias = ta.Generator[ta.Any, ta.Any, ta.Any]
|
17
17
|
|
18
18
|
|
@@ -93,8 +93,10 @@ class GenMachine(ta.Generic[I, O]):
|
|
93
93
|
if self._gen is None:
|
94
94
|
raise GenMachine.ClosedError
|
95
95
|
|
96
|
+
gi: I | None = i
|
96
97
|
try:
|
97
|
-
while (o := self._gen.send(
|
98
|
+
while (o := self._gen.send(gi)) is not None:
|
99
|
+
gi = None
|
98
100
|
yield from o
|
99
101
|
|
100
102
|
except StopIteration as s:
|
@@ -20,7 +20,7 @@ import codecs
|
|
20
20
|
import dataclasses as dc
|
21
21
|
import typing as ta
|
22
22
|
|
23
|
-
from
|
23
|
+
from .. import lang
|
24
24
|
|
25
25
|
|
26
26
|
if ta.TYPE_CHECKING:
|
@@ -39,7 +39,7 @@ if ta.TYPE_CHECKING:
|
|
39
39
|
import yaml as _yaml
|
40
40
|
import zstandard as _zstandard
|
41
41
|
|
42
|
-
from
|
42
|
+
from ..formats import json as _json
|
43
43
|
|
44
44
|
else:
|
45
45
|
_bz2 = lang.proxy_import('bz2')
|
@@ -57,7 +57,7 @@ else:
|
|
57
57
|
_yaml = lang.proxy_import('yaml')
|
58
58
|
_zstandard = lang.proxy_import('zstandard')
|
59
59
|
|
60
|
-
_json = lang.proxy_import('
|
60
|
+
_json = lang.proxy_import('..formats.json', __package__)
|
61
61
|
|
62
62
|
|
63
63
|
##
|
omlish/http/sessions.py
CHANGED
@@ -8,9 +8,9 @@ import time
|
|
8
8
|
import typing as ta
|
9
9
|
import zlib
|
10
10
|
|
11
|
-
from .. import fnpairs as fpa
|
12
11
|
from .. import lang
|
13
12
|
from .. import secrets as sec
|
13
|
+
from ..funcs import pairs as fpa
|
14
14
|
from .cookies import dump_cookie
|
15
15
|
from .cookies import parse_cookie
|
16
16
|
from .json import JSON_TAGGER
|
File without changes
|
@@ -0,0 +1,104 @@
|
|
1
|
+
"""
|
2
|
+
https://docs.python.org/3/library/bz2.html#bz2.BZ2Compressor
|
3
|
+
https://docs.python.org/3/library/zlib.html#zlib.decompressobj
|
4
|
+
https://docs.python.org/3/library/lzma.html#lzma.LZMADecompressor
|
5
|
+
"""
|
6
|
+
import abc
|
7
|
+
|
8
|
+
|
9
|
+
##
|
10
|
+
|
11
|
+
|
12
|
+
class Compressor(abc.ABC):
|
13
|
+
@abc.abstractmethod
|
14
|
+
def compress(self, data: bytes) -> bytes:
|
15
|
+
"""
|
16
|
+
Provide data to the compressor object. Returns a chunk of compressed data if possible, or an empty byte string
|
17
|
+
otherwise.
|
18
|
+
|
19
|
+
When you have finished providing data to the compressor, call the flush() method to finish the compression
|
20
|
+
process.
|
21
|
+
"""
|
22
|
+
|
23
|
+
raise NotImplementedError
|
24
|
+
|
25
|
+
@abc.abstractmethod
|
26
|
+
def flush(self) -> bytes:
|
27
|
+
"""
|
28
|
+
Finish the compression process. Returns the compressed data left in internal buffers.
|
29
|
+
|
30
|
+
The compressor object may not be used after this method has been called.
|
31
|
+
"""
|
32
|
+
|
33
|
+
raise NotImplementedError
|
34
|
+
|
35
|
+
|
36
|
+
##
|
37
|
+
|
38
|
+
|
39
|
+
class Decompressor(abc.ABC):
|
40
|
+
@property
|
41
|
+
@abc.abstractmethod
|
42
|
+
def unused_data(self) -> bytes:
|
43
|
+
"""
|
44
|
+
Data found after the end of the compressed stream.
|
45
|
+
|
46
|
+
If this attribute is accessed before the end of the stream has been reached, its value will be b''.
|
47
|
+
"""
|
48
|
+
|
49
|
+
raise NotImplementedError
|
50
|
+
|
51
|
+
@property
|
52
|
+
@abc.abstractmethod
|
53
|
+
def eof(self) -> bool:
|
54
|
+
"""True if the end-of-stream marker has been reached."""
|
55
|
+
|
56
|
+
raise NotImplementedError
|
57
|
+
|
58
|
+
@abc.abstractmethod
|
59
|
+
def decompress(self, data: bytes, *max_length: int) -> bytes:
|
60
|
+
"""
|
61
|
+
Decompress data, returning a bytes object containing the uncompressed data corresponding to at least part of the
|
62
|
+
data in string. This data should be concatenated to the output produced by any preceding calls to the
|
63
|
+
decompress() method. Some of the input data may be preserved in internal buffers for later processing.
|
64
|
+
|
65
|
+
If the optional parameter max_length is non-zero then the return value will be no longer than max_length.
|
66
|
+
"""
|
67
|
+
|
68
|
+
raise NotImplementedError
|
69
|
+
|
70
|
+
|
71
|
+
class NeedsInputDecompressor(Decompressor):
|
72
|
+
"""
|
73
|
+
Used by:
|
74
|
+
- bz2.BZ2Decompressor
|
75
|
+
- lzma.LZMADecompressor
|
76
|
+
"""
|
77
|
+
|
78
|
+
@property
|
79
|
+
@abc.abstractmethod
|
80
|
+
def needs_input(self) -> bool:
|
81
|
+
"""
|
82
|
+
False if the decompress() method can provide more decompressed data before requiring new uncompressed input.
|
83
|
+
"""
|
84
|
+
|
85
|
+
raise NotImplementedError
|
86
|
+
|
87
|
+
|
88
|
+
class UnconsumedTailDecompressor(Decompressor):
|
89
|
+
"""
|
90
|
+
Used by:
|
91
|
+
- zlib.decompressobj
|
92
|
+
"""
|
93
|
+
|
94
|
+
@property
|
95
|
+
@abc.abstractmethod
|
96
|
+
def unconsumed_tail(self) -> bytes:
|
97
|
+
"""
|
98
|
+
A bytes object that contains any data that was not consumed by the last decompress() call because it exceeded
|
99
|
+
the limit for the uncompressed data buffer. This data has not yet been seen by the zlib machinery, so you must
|
100
|
+
feed it (possibly with further data concatenated to it) back to a subsequent decompress() method call in order
|
101
|
+
to get correct output.
|
102
|
+
"""
|
103
|
+
|
104
|
+
raise NotImplementedError
|
@@ -0,0 +1,147 @@
|
|
1
|
+
# PYTHON SOFTWARE FOUNDATION LICENSE VERSION 2
|
2
|
+
# --------------------------------------------
|
3
|
+
#
|
4
|
+
# 1. This LICENSE AGREEMENT is between the Python Software Foundation ("PSF"), and the Individual or Organization
|
5
|
+
# ("Licensee") accessing and otherwise using this software ("Python") in source or binary form and its associated
|
6
|
+
# documentation.
|
7
|
+
#
|
8
|
+
# 2. Subject to the terms and conditions of this License Agreement, PSF hereby grants Licensee a nonexclusive,
|
9
|
+
# royalty-free, world-wide license to reproduce, analyze, test, perform and/or display publicly, prepare derivative
|
10
|
+
# works, distribute, and otherwise use Python alone or in any derivative version, provided, however, that PSF's License
|
11
|
+
# Agreement and PSF's notice of copyright, i.e., "Copyright (c) 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, 2009,
|
12
|
+
# 2010, 2011, 2012, 2013, 2014, 2015, 2016, 2017 Python Software Foundation; All Rights Reserved" are retained in Python
|
13
|
+
# alone or in any derivative version prepared by Licensee.
|
14
|
+
#
|
15
|
+
# 3. In the event Licensee prepares a derivative work that is based on or incorporates Python or any part thereof, and
|
16
|
+
# wants to make the derivative work available to others as provided herein, then Licensee hereby agrees to include in
|
17
|
+
# any such work a brief summary of the changes made to Python.
|
18
|
+
#
|
19
|
+
# 4. PSF is making Python available to Licensee on an "AS IS" basis. PSF MAKES NO REPRESENTATIONS OR WARRANTIES,
|
20
|
+
# EXPRESS OR IMPLIED. BY WAY OF EXAMPLE, BUT NOT LIMITATION, PSF MAKES NO AND DISCLAIMS ANY REPRESENTATION OR WARRANTY
|
21
|
+
# OF MERCHANTABILITY OR FITNESS FOR ANY PARTICULAR PURPOSE OR THAT THE USE OF PYTHON WILL NOT INFRINGE ANY THIRD PARTY
|
22
|
+
# RIGHTS.
|
23
|
+
#
|
24
|
+
# 5. PSF SHALL NOT BE LIABLE TO LICENSEE OR ANY OTHER USERS OF PYTHON FOR ANY INCIDENTAL, SPECIAL, OR CONSEQUENTIAL
|
25
|
+
# DAMAGES OR LOSS AS A RESULT OF MODIFYING, DISTRIBUTING, OR OTHERWISE USING PYTHON, OR ANY DERIVATIVE THEREOF, EVEN IF
|
26
|
+
# ADVISED OF THE POSSIBILITY THEREOF.
|
27
|
+
#
|
28
|
+
# 6. This License Agreement will automatically terminate upon a material breach of its terms and conditions.
|
29
|
+
#
|
30
|
+
# 7. Nothing in this License Agreement shall be deemed to create any relationship of agency, partnership, or joint
|
31
|
+
# venture between PSF and Licensee. This License Agreement does not grant permission to use PSF trademarks or trade
|
32
|
+
# name in a trademark sense to endorse or promote products or services of Licensee, or any third party.
|
33
|
+
#
|
34
|
+
# 8. By copying, installing or otherwise using Python, Licensee agrees to be bound by the terms and conditions of this
|
35
|
+
# License Agreement.
|
36
|
+
# ~> https://github.com/python/cpython/blob/f19c50a4817ffebb26132182ed8cec6a72342cc0/Lib/_compression.py
|
37
|
+
import typing as ta
|
38
|
+
|
39
|
+
from ... import check
|
40
|
+
from .abc import Compressor
|
41
|
+
from .abc import NeedsInputDecompressor
|
42
|
+
from .abc import UnconsumedTailDecompressor
|
43
|
+
from .types import IncrementalCompressor
|
44
|
+
from .types import IncrementalDecompressor
|
45
|
+
|
46
|
+
|
47
|
+
##
|
48
|
+
|
49
|
+
|
50
|
+
class CompressorIncrementalAdapter:
|
51
|
+
def __init__(
|
52
|
+
self,
|
53
|
+
factory: ta.Callable[..., Compressor],
|
54
|
+
) -> None:
|
55
|
+
super().__init__()
|
56
|
+
|
57
|
+
self._factory = factory
|
58
|
+
|
59
|
+
def __call__(self) -> IncrementalCompressor:
|
60
|
+
compressor = self._factory()
|
61
|
+
|
62
|
+
while True:
|
63
|
+
data = check.isinstance((yield None), bytes)
|
64
|
+
if not data:
|
65
|
+
break
|
66
|
+
|
67
|
+
compressed = compressor.compress(data)
|
68
|
+
if compressed:
|
69
|
+
check.none((yield compressed))
|
70
|
+
|
71
|
+
if (fl := compressor.flush()):
|
72
|
+
check.none((yield fl))
|
73
|
+
|
74
|
+
check.none((yield b''))
|
75
|
+
|
76
|
+
|
77
|
+
##
|
78
|
+
|
79
|
+
|
80
|
+
class DecompressorIncrementalAdapter:
|
81
|
+
def __init__(
|
82
|
+
self,
|
83
|
+
factory: ta.Callable[..., NeedsInputDecompressor | UnconsumedTailDecompressor],
|
84
|
+
*,
|
85
|
+
trailing_error: type[BaseException] | tuple[type[BaseException], ...] = (),
|
86
|
+
) -> None:
|
87
|
+
super().__init__()
|
88
|
+
|
89
|
+
self._factory = factory
|
90
|
+
self._trailing_error = trailing_error
|
91
|
+
|
92
|
+
def __call__(self) -> IncrementalDecompressor:
|
93
|
+
pos = 0
|
94
|
+
|
95
|
+
data = None # Default if EOF is encountered
|
96
|
+
|
97
|
+
decompressor = self._factory()
|
98
|
+
|
99
|
+
while True:
|
100
|
+
# Depending on the input data, our call to the decompressor may not return any data. In this case, try again
|
101
|
+
# after reading another block.
|
102
|
+
while True:
|
103
|
+
if decompressor.eof:
|
104
|
+
rawblock = decompressor.unused_data
|
105
|
+
if not rawblock:
|
106
|
+
rawblock = check.isinstance((yield None), bytes)
|
107
|
+
if not rawblock:
|
108
|
+
break
|
109
|
+
|
110
|
+
# Continue to next stream.
|
111
|
+
decompressor = self._factory()
|
112
|
+
|
113
|
+
try:
|
114
|
+
data = decompressor.decompress(rawblock)
|
115
|
+
except self._trailing_error:
|
116
|
+
# Trailing data isn't a valid compressed stream; ignore it.
|
117
|
+
break
|
118
|
+
|
119
|
+
else:
|
120
|
+
if hasattr(decompressor, 'needs_input'):
|
121
|
+
if decompressor.needs_input:
|
122
|
+
rawblock = check.isinstance((yield None), bytes)
|
123
|
+
if not rawblock:
|
124
|
+
raise EOFError('Compressed file ended before the end-of-stream marker was reached')
|
125
|
+
else:
|
126
|
+
rawblock = b''
|
127
|
+
|
128
|
+
elif hasattr(decompressor, 'unconsumed_tail'):
|
129
|
+
if not (rawblock := decompressor.unconsumed_tail):
|
130
|
+
rawblock = check.isinstance((yield None), bytes)
|
131
|
+
if not rawblock:
|
132
|
+
raise EOFError('Compressed file ended before the end-of-stream marker was reached')
|
133
|
+
|
134
|
+
else:
|
135
|
+
raise TypeError(decompressor)
|
136
|
+
|
137
|
+
data = decompressor.decompress(rawblock)
|
138
|
+
|
139
|
+
if data:
|
140
|
+
break
|
141
|
+
|
142
|
+
if not data:
|
143
|
+
check.none((yield b''))
|
144
|
+
return
|
145
|
+
|
146
|
+
pos += len(data)
|
147
|
+
check.none((yield data))
|
@@ -0,0 +1,41 @@
|
|
1
|
+
import functools
|
2
|
+
import typing as ta
|
3
|
+
|
4
|
+
from ... import lang
|
5
|
+
from .adapters import CompressorIncrementalAdapter
|
6
|
+
from .adapters import DecompressorIncrementalAdapter
|
7
|
+
from .types import IncrementalCompressor
|
8
|
+
from .types import IncrementalDecompressor
|
9
|
+
|
10
|
+
|
11
|
+
if ta.TYPE_CHECKING:
|
12
|
+
import bz2
|
13
|
+
else:
|
14
|
+
bz2 = lang.proxy_import('bz2')
|
15
|
+
|
16
|
+
|
17
|
+
class IncrementalBz2Compressor:
|
18
|
+
def __init__(
|
19
|
+
self,
|
20
|
+
*,
|
21
|
+
compresslevel: int = 9,
|
22
|
+
) -> None:
|
23
|
+
super().__init__()
|
24
|
+
|
25
|
+
self._compresslevel = compresslevel
|
26
|
+
|
27
|
+
def __call__(self) -> IncrementalCompressor:
|
28
|
+
return CompressorIncrementalAdapter(
|
29
|
+
functools.partial(
|
30
|
+
bz2.BZ2Compressor, # type: ignore
|
31
|
+
self._compresslevel,
|
32
|
+
),
|
33
|
+
)()
|
34
|
+
|
35
|
+
|
36
|
+
class IncrementalBz2Decompressor:
|
37
|
+
def __call__(self) -> IncrementalDecompressor:
|
38
|
+
return DecompressorIncrementalAdapter(
|
39
|
+
bz2.BZ2Decompressor, # type: ignore
|
40
|
+
trailing_error=OSError,
|
41
|
+
)()
|