omlish 0.0.0.dev253__py3-none-any.whl → 0.0.0.dev255__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.
- omlish/__about__.py +2 -2
- omlish/dispatch/impls.py +3 -2
- omlish/lang/objects.py +23 -6
- omlish/os/filemodes.py +1 -1
- omlish/sql/parsing/Minisql.g4 +8 -5
- omlish/subprocesses/wrap.py +5 -0
- omlish/text/decoding.py +73 -0
- omlish/text/filecache.py +235 -0
- omlish/text/linecache.py +80 -0
- omlish/text/parts.py +58 -44
- {omlish-0.0.0.dev253.dist-info → omlish-0.0.0.dev255.dist-info}/METADATA +1 -1
- {omlish-0.0.0.dev253.dist-info → omlish-0.0.0.dev255.dist-info}/RECORD +16 -13
- {omlish-0.0.0.dev253.dist-info → omlish-0.0.0.dev255.dist-info}/WHEEL +1 -1
- {omlish-0.0.0.dev253.dist-info → omlish-0.0.0.dev255.dist-info}/LICENSE +0 -0
- {omlish-0.0.0.dev253.dist-info → omlish-0.0.0.dev255.dist-info}/entry_points.txt +0 -0
- {omlish-0.0.0.dev253.dist-info → omlish-0.0.0.dev255.dist-info}/top_level.txt +0 -0
omlish/__about__.py
CHANGED
omlish/dispatch/impls.py
CHANGED
@@ -5,7 +5,6 @@ TODO:
|
|
5
5
|
- multidispatch? never solved..
|
6
6
|
- just generic on tuple[A0, A1, ...]
|
7
7
|
"""
|
8
|
-
import contextlib
|
9
8
|
import typing as ta
|
10
9
|
import weakref
|
11
10
|
|
@@ -24,8 +23,10 @@ _IMPL_FUNC_CLS_SET_CACHE: ta.MutableMapping[ta.Callable, frozenset[type]] = weak
|
|
24
23
|
|
25
24
|
|
26
25
|
def get_impl_func_cls_set(func: ta.Callable) -> frozenset[type]:
|
27
|
-
|
26
|
+
try:
|
28
27
|
return _IMPL_FUNC_CLS_SET_CACHE[func]
|
28
|
+
except KeyError:
|
29
|
+
pass
|
29
30
|
|
30
31
|
ann = getattr(func, '__annotations__', {})
|
31
32
|
if not ann:
|
omlish/lang/objects.py
CHANGED
@@ -11,11 +11,24 @@ T = ta.TypeVar('T')
|
|
11
11
|
##
|
12
12
|
|
13
13
|
|
14
|
-
def attr_repr(
|
15
|
-
|
14
|
+
def attr_repr(
|
15
|
+
obj: ta.Any,
|
16
|
+
*attrs: str,
|
17
|
+
with_module: bool = False,
|
18
|
+
use_qualname: bool = False,
|
19
|
+
with_id: bool = False,
|
20
|
+
) -> str:
|
21
|
+
return (
|
22
|
+
f'{obj.__class__.__module__ + "." if with_module else ""}'
|
23
|
+
f'{obj.__class__.__qualname__ if use_qualname else obj.__class__.__name__}'
|
24
|
+
f'{("@" + hex(id(obj))[2:]) if with_id else ""}'
|
25
|
+
f'('
|
26
|
+
f'{", ".join(f"{attr}={getattr(obj, attr)!r}" for attr in attrs)}'
|
27
|
+
f')'
|
28
|
+
)
|
16
29
|
|
17
30
|
|
18
|
-
def arg_repr(*args, **kwargs) -> str:
|
31
|
+
def arg_repr(*args: ta.Any, **kwargs: ta.Any) -> str:
|
19
32
|
return ', '.join(*(
|
20
33
|
list(map(repr, args)) +
|
21
34
|
[f'{k}={v!r}' for k, v in kwargs.items()]
|
@@ -57,7 +70,7 @@ def new_type(
|
|
57
70
|
name: str,
|
58
71
|
bases: ta.Sequence[ta.Any],
|
59
72
|
namespace: ta.Mapping[str, ta.Any],
|
60
|
-
**kwargs,
|
73
|
+
**kwargs: ta.Any,
|
61
74
|
) -> type:
|
62
75
|
return types.new_class(
|
63
76
|
name,
|
@@ -73,7 +86,7 @@ def super_meta(
|
|
73
86
|
name: str,
|
74
87
|
bases: ta.Sequence[ta.Any],
|
75
88
|
namespace: ta.MutableMapping[str, ta.Any],
|
76
|
-
**kwargs,
|
89
|
+
**kwargs: ta.Any,
|
77
90
|
) -> type:
|
78
91
|
"""Per types.new_class"""
|
79
92
|
resolved_bases = types.resolve_bases(bases)
|
@@ -87,7 +100,11 @@ def super_meta(
|
|
87
100
|
##
|
88
101
|
|
89
102
|
|
90
|
-
def deep_subclasses(
|
103
|
+
def deep_subclasses(
|
104
|
+
cls: type[T],
|
105
|
+
*,
|
106
|
+
concrete_only: bool = False,
|
107
|
+
) -> ta.Iterator[type[T]]:
|
91
108
|
seen = set()
|
92
109
|
todo = list(reversed(cls.__subclasses__()))
|
93
110
|
while todo:
|
omlish/os/filemodes.py
CHANGED
@@ -24,7 +24,7 @@ class FileMode:
|
|
24
24
|
|
25
25
|
https://en.cppreference.com/w/cpp/io/c/fopen
|
26
26
|
|
27
|
-
"r" read Open a file for reading
|
27
|
+
"r" read Open a file for reading read from start return NULL and set error
|
28
28
|
"w" write Create a file for writing destroy contents create new
|
29
29
|
"a" append Append to a file write to end create new
|
30
30
|
"r+" read extended Open a file for read/write read from start return NULL and set error
|
omlish/sql/parsing/Minisql.g4
CHANGED
@@ -100,11 +100,14 @@ sortItem
|
|
100
100
|
;
|
101
101
|
|
102
102
|
relation
|
103
|
-
: relation AS? ident
|
104
|
-
| left=relation
|
105
|
-
|
106
|
-
|
107
|
-
|
103
|
+
: relation AS? ident #aliasedRelation
|
104
|
+
| left=relation
|
105
|
+
ty=joinType?
|
106
|
+
JOIN right=relation
|
107
|
+
(ON cond=booleanExpr)? #joinRelation
|
108
|
+
| '(' select ')' #selectRelation
|
109
|
+
| '(' relation ')' #parenRelation
|
110
|
+
| qualifiedName #tableRelation
|
108
111
|
;
|
109
112
|
|
110
113
|
groupBy
|
omlish/subprocesses/wrap.py
CHANGED
@@ -1,5 +1,10 @@
|
|
1
1
|
# ruff: noqa: UP006 UP007
|
2
2
|
# @omlish-lite
|
3
|
+
"""
|
4
|
+
This bypasses debuggers attaching to spawned subprocess children that look like python processes. See:
|
5
|
+
|
6
|
+
https://github.com/JetBrains/intellij-community/blob/e9d8f126c286acf9df3ff272f440b305bf2ff585/python/helpers/pydev/_pydev_bundle/pydev_monkey.py
|
7
|
+
"""
|
3
8
|
import shlex
|
4
9
|
import typing as ta
|
5
10
|
|
omlish/text/decoding.py
ADDED
@@ -0,0 +1,73 @@
|
|
1
|
+
"""
|
2
|
+
TODO:
|
3
|
+
- cext
|
4
|
+
"""
|
5
|
+
import array
|
6
|
+
import codecs
|
7
|
+
import typing as ta
|
8
|
+
|
9
|
+
|
10
|
+
##
|
11
|
+
|
12
|
+
|
13
|
+
def decode_to_list(
|
14
|
+
raw: bytes,
|
15
|
+
encoding: str = 'utf-8',
|
16
|
+
*,
|
17
|
+
errors: str = 'strict',
|
18
|
+
) -> list[str]:
|
19
|
+
dec = codecs.getincrementaldecoder(encoding)(errors)
|
20
|
+
end = len(raw) - 1
|
21
|
+
return [dec.decode(bytes([b]), final=i == end) for i, b in enumerate(raw)]
|
22
|
+
|
23
|
+
|
24
|
+
#
|
25
|
+
|
26
|
+
|
27
|
+
class DecodedStringIndex(ta.NamedTuple):
|
28
|
+
byte_offsets: ta.Sequence[int]
|
29
|
+
|
30
|
+
|
31
|
+
def decode_indexed(
|
32
|
+
raw: bytes,
|
33
|
+
encoding: str = 'utf-8',
|
34
|
+
*,
|
35
|
+
errors: str = 'strict',
|
36
|
+
) -> tuple[str, DecodedStringIndex]:
|
37
|
+
dec_lst = decode_to_list(
|
38
|
+
raw,
|
39
|
+
encoding,
|
40
|
+
errors=errors,
|
41
|
+
)
|
42
|
+
|
43
|
+
dec_s = ''.join(dec_lst)
|
44
|
+
|
45
|
+
bo_arr_len = len(dec_s) + 1
|
46
|
+
if bo_arr_len < 2**8:
|
47
|
+
fmt = 'B'
|
48
|
+
elif bo_arr_len < 2**16:
|
49
|
+
fmt = 'H'
|
50
|
+
elif bo_arr_len < 2**32:
|
51
|
+
fmt = 'L'
|
52
|
+
else:
|
53
|
+
fmt = 'Q'
|
54
|
+
|
55
|
+
bo_arr = array.array(fmt, [0]) * bo_arr_len
|
56
|
+
so = 0
|
57
|
+
lbo = 0
|
58
|
+
for bo, s in enumerate(dec_lst):
|
59
|
+
if s:
|
60
|
+
bo_arr[so] = lbo
|
61
|
+
so += 1
|
62
|
+
lbo = bo + 1
|
63
|
+
|
64
|
+
if so != len(dec_s):
|
65
|
+
raise RuntimeError
|
66
|
+
|
67
|
+
bo_arr[len(dec_s)] = len(raw)
|
68
|
+
|
69
|
+
dsi = DecodedStringIndex(
|
70
|
+
bo_arr,
|
71
|
+
)
|
72
|
+
|
73
|
+
return (dec_s, dsi)
|
omlish/text/filecache.py
ADDED
@@ -0,0 +1,235 @@
|
|
1
|
+
"""
|
2
|
+
Features:
|
3
|
+
- get (line, offset)
|
4
|
+
- invalidate
|
5
|
+
- filewatcher
|
6
|
+
- linecache interop
|
7
|
+
- tokenize.open - detect encoding - only for .py
|
8
|
+
- directory / path_parts tree?
|
9
|
+
|
10
|
+
TODO:
|
11
|
+
- read raw, decode (detect)
|
12
|
+
- ! index from byteofs -> charofs
|
13
|
+
- charofs -> lineno
|
14
|
+
- 1-based?
|
15
|
+
- maximums?
|
16
|
+
- lru? max_size/max_entries?
|
17
|
+
- collections.cache?
|
18
|
+
|
19
|
+
Notes:
|
20
|
+
- linecache is 1-based
|
21
|
+
- linecache appends final newline if missing:
|
22
|
+
if lines and not lines[-1].endswith('\n'):
|
23
|
+
lines[-1] += '\n'
|
24
|
+
"""
|
25
|
+
import dataclasses as dc
|
26
|
+
import io
|
27
|
+
import os.path
|
28
|
+
import stat as stat_
|
29
|
+
import tokenize
|
30
|
+
import typing as ta
|
31
|
+
|
32
|
+
from .. import cached
|
33
|
+
from .. import lang
|
34
|
+
from ..os.paths import abs_real_path
|
35
|
+
|
36
|
+
|
37
|
+
##
|
38
|
+
|
39
|
+
|
40
|
+
class TextFileCache:
|
41
|
+
class FileStat(ta.NamedTuple):
|
42
|
+
size: int
|
43
|
+
mtime: float
|
44
|
+
|
45
|
+
class Entry:
|
46
|
+
def __init__(
|
47
|
+
self,
|
48
|
+
cache: 'TextFileCache',
|
49
|
+
path: str,
|
50
|
+
stat: 'TextFileCache.FileStat',
|
51
|
+
*,
|
52
|
+
encoding: str | None = None,
|
53
|
+
) -> None:
|
54
|
+
super().__init__()
|
55
|
+
|
56
|
+
self._cache = cache
|
57
|
+
self._path = path
|
58
|
+
self._stat = stat
|
59
|
+
self._given_encoding = encoding
|
60
|
+
|
61
|
+
def __repr__(self) -> str:
|
62
|
+
return lang.attr_repr(self, 'path', 'stat')
|
63
|
+
|
64
|
+
@property
|
65
|
+
def path(self) -> str:
|
66
|
+
return self._path
|
67
|
+
|
68
|
+
@property
|
69
|
+
def stat(self) -> 'TextFileCache.FileStat':
|
70
|
+
return self._stat
|
71
|
+
|
72
|
+
@cached.function
|
73
|
+
def raw(self) -> bytes:
|
74
|
+
return self._cache._read_file(self._path) # noqa
|
75
|
+
|
76
|
+
@cached.function
|
77
|
+
def encoding(self) -> str:
|
78
|
+
if (ge := self._given_encoding) is not None:
|
79
|
+
return ge
|
80
|
+
return self._cache._determine_encoding(self._path, self.raw()) # noqa
|
81
|
+
|
82
|
+
@cached.function
|
83
|
+
def text(self) -> str:
|
84
|
+
return self.raw().decode(self.encoding())
|
85
|
+
|
86
|
+
@cached.function
|
87
|
+
def lines(self) -> ta.Sequence[str]:
|
88
|
+
return self.text().splitlines(keepends=True)
|
89
|
+
|
90
|
+
def safe_line(self, n: int, default: str = '') -> str:
|
91
|
+
lines = self.lines()
|
92
|
+
if 0 <= n < len(lines):
|
93
|
+
return lines[n]
|
94
|
+
return default
|
95
|
+
|
96
|
+
#
|
97
|
+
|
98
|
+
@dc.dataclass(frozen=True)
|
99
|
+
class Config:
|
100
|
+
max_file_size: int | None = 5 * 1024 * 1024
|
101
|
+
|
102
|
+
default_encoding: str = 'utf-8'
|
103
|
+
|
104
|
+
def __init__(
|
105
|
+
self,
|
106
|
+
config: Config = Config(),
|
107
|
+
*,
|
108
|
+
locking: lang.DefaultLockable = None,
|
109
|
+
) -> None:
|
110
|
+
super().__init__()
|
111
|
+
|
112
|
+
self._config = config
|
113
|
+
|
114
|
+
self._dct: dict[str, TextFileCache.Entry] = {}
|
115
|
+
self._lock = lang.default_lock(locking)()
|
116
|
+
|
117
|
+
#
|
118
|
+
|
119
|
+
class Error(Exception):
|
120
|
+
pass
|
121
|
+
|
122
|
+
@dc.dataclass()
|
123
|
+
class PathError(Error):
|
124
|
+
path: str
|
125
|
+
|
126
|
+
class PathTypeError(PathError):
|
127
|
+
pass
|
128
|
+
|
129
|
+
class FileTooBigError(PathError):
|
130
|
+
pass
|
131
|
+
|
132
|
+
#
|
133
|
+
|
134
|
+
def _normalize_path(self, path: str) -> str:
|
135
|
+
return abs_real_path(path)
|
136
|
+
|
137
|
+
#
|
138
|
+
|
139
|
+
def _try_stat_file(self, path: str) -> FileStat | Exception:
|
140
|
+
try:
|
141
|
+
st = os.stat(path)
|
142
|
+
except FileNotFoundError as e:
|
143
|
+
return e
|
144
|
+
|
145
|
+
if not stat_.S_ISREG(st.st_mode):
|
146
|
+
return TextFileCache.PathTypeError(path)
|
147
|
+
|
148
|
+
return TextFileCache.FileStat(
|
149
|
+
st.st_size,
|
150
|
+
st.st_mtime,
|
151
|
+
)
|
152
|
+
|
153
|
+
def _read_file(self, path: str) -> bytes:
|
154
|
+
with open(path, 'rb') as f:
|
155
|
+
return f.read()
|
156
|
+
|
157
|
+
def _determine_encoding(self, path: str, raw: bytes) -> str:
|
158
|
+
if path.endswith('.py'):
|
159
|
+
buf = io.BytesIO(raw)
|
160
|
+
encoding, _ = tokenize.detect_encoding(buf.readline)
|
161
|
+
return encoding
|
162
|
+
|
163
|
+
return self._config.default_encoding
|
164
|
+
|
165
|
+
#
|
166
|
+
|
167
|
+
def _get_entry(
|
168
|
+
self,
|
169
|
+
path: str,
|
170
|
+
*,
|
171
|
+
check_stat: bool = False,
|
172
|
+
or_raise: bool = False,
|
173
|
+
) -> Entry:
|
174
|
+
st_: TextFileCache.FileStat | Exception | None = None
|
175
|
+
try:
|
176
|
+
e = self._dct[path]
|
177
|
+
|
178
|
+
except KeyError:
|
179
|
+
if or_raise:
|
180
|
+
raise
|
181
|
+
|
182
|
+
else:
|
183
|
+
if (
|
184
|
+
not check_stat or
|
185
|
+
(st_ := self._try_stat_file(path)) == e.stat
|
186
|
+
):
|
187
|
+
return e
|
188
|
+
|
189
|
+
del self._dct[path]
|
190
|
+
|
191
|
+
st: TextFileCache.FileStat | Exception
|
192
|
+
if st_ is not None:
|
193
|
+
st = st_
|
194
|
+
else:
|
195
|
+
st = self._try_stat_file(path)
|
196
|
+
|
197
|
+
if isinstance(st, Exception):
|
198
|
+
raise st
|
199
|
+
|
200
|
+
if (mfs := self._config.max_file_size) is not None and st.size > mfs:
|
201
|
+
raise TextFileCache.FileTooBigError(path)
|
202
|
+
|
203
|
+
e = TextFileCache.Entry(
|
204
|
+
self,
|
205
|
+
path,
|
206
|
+
st,
|
207
|
+
)
|
208
|
+
|
209
|
+
self._dct[path] = e
|
210
|
+
|
211
|
+
return e
|
212
|
+
|
213
|
+
def get_entry(
|
214
|
+
self,
|
215
|
+
path: str,
|
216
|
+
*,
|
217
|
+
check_stat: bool = False,
|
218
|
+
or_raise: bool = False,
|
219
|
+
) -> Entry:
|
220
|
+
path = self._normalize_path(path)
|
221
|
+
|
222
|
+
with self._lock:
|
223
|
+
return self._get_entry(
|
224
|
+
path,
|
225
|
+
check_stat=check_stat,
|
226
|
+
or_raise=or_raise,
|
227
|
+
)
|
228
|
+
|
229
|
+
#
|
230
|
+
|
231
|
+
def invalidate(self, path: str) -> bool:
|
232
|
+
path = self._normalize_path(path)
|
233
|
+
|
234
|
+
with self._lock:
|
235
|
+
return self._dct.pop(path, None) is not None
|
omlish/text/linecache.py
ADDED
@@ -0,0 +1,80 @@
|
|
1
|
+
"""
|
2
|
+
TODO:
|
3
|
+
- reserver / generator
|
4
|
+
|
5
|
+
==
|
6
|
+
|
7
|
+
@dc.dataclass()
|
8
|
+
class _ReservedFilenameEntry:
|
9
|
+
unique_id: str
|
10
|
+
seq: int = 0
|
11
|
+
|
12
|
+
|
13
|
+
_RESERVED_FILENAME_UUID_THREAD_LOCAL = threading.local()
|
14
|
+
|
15
|
+
|
16
|
+
def reserve_filename(prefix: str) -> str:
|
17
|
+
try:
|
18
|
+
e = _RESERVED_FILENAME_UUID_THREAD_LOCAL.unique_id
|
19
|
+
except AttributeError:
|
20
|
+
e = _RESERVED_FILENAME_UUID_THREAD_LOCAL.unique_id = _ReservedFilenameEntry(str(uuid.uuid4()))
|
21
|
+
while True:
|
22
|
+
unique_filename = f'<generated:{prefix}:{e.seq}>'
|
23
|
+
cache_line = (1, None, (e.unique_id,), unique_filename)
|
24
|
+
e.seq += 1
|
25
|
+
if linecache.cache.setdefault(unique_filename, cache_line) == cache_line:
|
26
|
+
return unique_filename
|
27
|
+
"""
|
28
|
+
import typing as ta
|
29
|
+
|
30
|
+
|
31
|
+
##
|
32
|
+
|
33
|
+
|
34
|
+
class LinecacheKey(ta.NamedTuple):
|
35
|
+
size: int # os.stat().st_size
|
36
|
+
mtime: float | None # os.stat().st_mtime
|
37
|
+
lines: ta.Sequence[str]
|
38
|
+
fullname: str
|
39
|
+
|
40
|
+
|
41
|
+
LinecacheLazyLoader: ta.TypeAlias = tuple[ta.Callable[[], str]]
|
42
|
+
|
43
|
+
LinecacheEntry: ta.TypeAlias = LinecacheKey | LinecacheLazyLoader
|
44
|
+
|
45
|
+
|
46
|
+
class LinecacheProtocol(ta.Protocol):
|
47
|
+
@property
|
48
|
+
def cache(self) -> ta.MutableMapping[str, LinecacheEntry]: ...
|
49
|
+
|
50
|
+
def clearcache(self) -> None: ...
|
51
|
+
|
52
|
+
def getline(
|
53
|
+
self,
|
54
|
+
filename: str,
|
55
|
+
lineno: int,
|
56
|
+
module_globals: ta.Mapping[str, ta.Any] | None = None,
|
57
|
+
) -> str: ...
|
58
|
+
|
59
|
+
def getlines(
|
60
|
+
self,
|
61
|
+
filename: str,
|
62
|
+
module_globals: ta.Mapping[str, ta.Any] | None = None,
|
63
|
+
) -> ta.Sequence[str]: ...
|
64
|
+
|
65
|
+
def checkcache(
|
66
|
+
self,
|
67
|
+
filename: str | None = None,
|
68
|
+
) -> None: ...
|
69
|
+
|
70
|
+
def updatecache(
|
71
|
+
self,
|
72
|
+
filename: str,
|
73
|
+
module_globals: ta.Mapping[str, ta.Any] | None = None,
|
74
|
+
) -> ta.Sequence[str]: ...
|
75
|
+
|
76
|
+
def lazycache(
|
77
|
+
self,
|
78
|
+
filename: str,
|
79
|
+
module_globals: ta.Mapping[str, ta.Any],
|
80
|
+
) -> bool: ...
|
omlish/text/parts.py
CHANGED
@@ -72,40 +72,47 @@ class Meta(DataPart):
|
|
72
72
|
|
73
73
|
|
74
74
|
class PartTransform:
|
75
|
+
def __call__(self, part: Part | None) -> Part:
|
76
|
+
return self._transform(part)
|
77
|
+
|
75
78
|
@dispatch.method
|
76
|
-
def
|
79
|
+
def _transform(self, part: Part | None) -> Part:
|
77
80
|
raise TypeError(part)
|
78
81
|
|
79
|
-
@
|
80
|
-
def
|
82
|
+
@_transform.register
|
83
|
+
def _transform_none(self, part: None) -> Part:
|
84
|
+
return []
|
85
|
+
|
86
|
+
@_transform.register
|
87
|
+
def _transform_str(self, part: str) -> Part:
|
81
88
|
return part
|
82
89
|
|
83
|
-
@
|
84
|
-
def
|
90
|
+
@_transform.register
|
91
|
+
def _transform_sequence(self, part: collections.abc.Sequence) -> Part:
|
85
92
|
return [self(c) for c in part]
|
86
93
|
|
87
|
-
@
|
88
|
-
def
|
94
|
+
@_transform.register
|
95
|
+
def _transform_wrap(self, part: Wrap) -> Part:
|
89
96
|
return Wrap(self(part.part), part.wrapper)
|
90
97
|
|
91
|
-
@
|
92
|
-
def
|
98
|
+
@_transform.register
|
99
|
+
def _transform_list(self, part: List) -> Part:
|
93
100
|
return List([self(c) for c in part.parts], part.delimiter, part.trailer)
|
94
101
|
|
95
|
-
@
|
96
|
-
def
|
102
|
+
@_transform.register
|
103
|
+
def _transform_concat(self, part: Concat) -> Part:
|
97
104
|
return Concat([self(c) for c in part.parts])
|
98
105
|
|
99
|
-
@
|
100
|
-
def
|
106
|
+
@_transform.register
|
107
|
+
def _transform_block(self, part: Block) -> Part:
|
101
108
|
return Block([self(c) for c in part.parts])
|
102
109
|
|
103
|
-
@
|
104
|
-
def
|
110
|
+
@_transform.register
|
111
|
+
def _transform_section(self, part: Section) -> Part:
|
105
112
|
return Section([self(c) for c in part.parts])
|
106
113
|
|
107
|
-
@
|
108
|
-
def
|
114
|
+
@_transform.register
|
115
|
+
def _transform_meta(self, part: Meta) -> Meta:
|
109
116
|
return part
|
110
117
|
|
111
118
|
|
@@ -113,8 +120,8 @@ class PartTransform:
|
|
113
120
|
|
114
121
|
|
115
122
|
class RemoveMetas(PartTransform):
|
116
|
-
@PartTransform.
|
117
|
-
def
|
123
|
+
@PartTransform._transform.register # noqa
|
124
|
+
def _transform_meta(self, part: Meta) -> Part:
|
118
125
|
return []
|
119
126
|
|
120
127
|
|
@@ -137,27 +144,27 @@ def _drop_empties(it: ta.Iterable[T]) -> list[T]:
|
|
137
144
|
|
138
145
|
|
139
146
|
class CompactPart(PartTransform):
|
140
|
-
@PartTransform.
|
141
|
-
def
|
147
|
+
@PartTransform._transform.register # noqa
|
148
|
+
def _transform_sequence(self, part: collections.abc.Sequence) -> Part:
|
142
149
|
return _drop_empties(self(c) for c in part)
|
143
150
|
|
144
|
-
@PartTransform.
|
145
|
-
def
|
151
|
+
@PartTransform._transform.register # noqa
|
152
|
+
def _transform_list(self, part: List) -> Part:
|
146
153
|
parts = _drop_empties(self(c) for c in part.parts)
|
147
154
|
return List(parts, part.delimiter, part.trailer) if parts else []
|
148
155
|
|
149
|
-
@PartTransform.
|
150
|
-
def
|
156
|
+
@PartTransform._transform.register # noqa
|
157
|
+
def _transform_concat(self, part: Concat) -> Part:
|
151
158
|
parts = _drop_empties(self(c) for c in part.parts)
|
152
159
|
return Concat(parts) if parts else []
|
153
160
|
|
154
|
-
@PartTransform.
|
155
|
-
def
|
161
|
+
@PartTransform._transform.register # noqa
|
162
|
+
def _transform_block(self, part: Block) -> Part:
|
156
163
|
parts = _drop_empties(self(c) for c in part.parts)
|
157
164
|
return Block(parts) if parts else []
|
158
165
|
|
159
|
-
@PartTransform.
|
160
|
-
def
|
166
|
+
@PartTransform._transform.register # noqa
|
167
|
+
def _transform_section(self, part: Section) -> Part:
|
161
168
|
parts = _drop_empties(self(c) for c in part.parts)
|
162
169
|
return Section(parts) if parts else []
|
163
170
|
|
@@ -204,29 +211,36 @@ class PartRenderer:
|
|
204
211
|
self._blank_lines += n
|
205
212
|
self._has_indented = False
|
206
213
|
|
214
|
+
def __call__(self, part: Part | None) -> None:
|
215
|
+
return self._render(part)
|
216
|
+
|
207
217
|
@dispatch.method
|
208
|
-
def
|
218
|
+
def _render(self, part: Part | None) -> None:
|
209
219
|
raise TypeError(part)
|
210
220
|
|
211
|
-
@
|
212
|
-
def
|
221
|
+
@_render.register
|
222
|
+
def _render_none(self, part: None) -> None:
|
223
|
+
pass
|
224
|
+
|
225
|
+
@_render.register
|
226
|
+
def _render_str(self, part: str) -> None:
|
213
227
|
self._write(part)
|
214
228
|
|
215
|
-
@
|
216
|
-
def
|
229
|
+
@_render.register
|
230
|
+
def _render_sequence(self, part: collections.abc.Sequence) -> None:
|
217
231
|
for i, c in enumerate(part):
|
218
232
|
if i:
|
219
233
|
self._write(' ')
|
220
234
|
self(c)
|
221
235
|
|
222
|
-
@
|
223
|
-
def
|
236
|
+
@_render.register
|
237
|
+
def _render_wrap(self, part: Wrap) -> None:
|
224
238
|
self._write(part.wrapper[0])
|
225
239
|
self(part.part)
|
226
240
|
self._write(part.wrapper[1])
|
227
241
|
|
228
|
-
@
|
229
|
-
def
|
242
|
+
@_render.register
|
243
|
+
def _render_list(self, part: List) -> None:
|
230
244
|
for i, c in enumerate(part.parts):
|
231
245
|
if i:
|
232
246
|
self._write(part.delimiter + ' ')
|
@@ -234,19 +248,19 @@ class PartRenderer:
|
|
234
248
|
if part.trailer:
|
235
249
|
self._write(part.delimiter)
|
236
250
|
|
237
|
-
@
|
238
|
-
def
|
251
|
+
@_render.register
|
252
|
+
def _render_concat(self, part: Concat) -> None:
|
239
253
|
for c in part.parts:
|
240
254
|
self(c)
|
241
255
|
|
242
|
-
@
|
243
|
-
def
|
256
|
+
@_render.register
|
257
|
+
def _render_block(self, part: Block) -> None:
|
244
258
|
for c in part.parts:
|
245
259
|
self(c)
|
246
260
|
self._write_newline()
|
247
261
|
|
248
|
-
@
|
249
|
-
def
|
262
|
+
@_render.register
|
263
|
+
def _render_section(self, part: Section) -> None:
|
250
264
|
self._indents += 1
|
251
265
|
try:
|
252
266
|
for c in part.parts:
|
@@ -1,5 +1,5 @@
|
|
1
1
|
omlish/.manifests.json,sha256=x26AIwDzScUvnX-p4xlq6Zc5QYrAo0Vmgf1qHc1KL_M,8253
|
2
|
-
omlish/__about__.py,sha256=
|
2
|
+
omlish/__about__.py,sha256=RWHoA3srEenVlQkvR7jq1Ub8ehSFGPwTcfcQOXal8Rk,3380
|
3
3
|
omlish/__init__.py,sha256=SsyiITTuK0v74XpKV8dqNaCmjOlan1JZKrHQv5rWKPA,253
|
4
4
|
omlish/c3.py,sha256=ubu7lHwss5V4UznbejAI0qXhXahrU01MysuHOZI9C4U,8116
|
5
5
|
omlish/cached.py,sha256=MLap_p0rdGoDIMVhXVHm1tsbcWobJF0OanoodV03Ju8,542
|
@@ -238,7 +238,7 @@ omlish/dispatch/_dispatch2.py,sha256=70k1tKKvuhxtAu6v4skECfHKIKVWrmlt7G_JKLUsKEs
|
|
238
238
|
omlish/dispatch/_dispatch3.py,sha256=9Zjd7bINAC3keiaBdssc4v5dY0-8OI6XooV2DR9U7Z0,2818
|
239
239
|
omlish/dispatch/dispatch.py,sha256=KA5l49AiGLRjp4J7RDJW9RiDp9WUD1ewR1AOPEF8g38,3062
|
240
240
|
omlish/dispatch/functions.py,sha256=8Qvj62XKn9SKfiqoVQdBD3wVRE4JUWpZDqs0oouuDIw,1519
|
241
|
-
omlish/dispatch/impls.py,sha256=
|
241
|
+
omlish/dispatch/impls.py,sha256=JP67zSvDWG4XFOmKPrhWHxRV9WnyAPAsjQHvbmEJh4c,2136
|
242
242
|
omlish/dispatch/methods.py,sha256=dr8xQo-zVRckIo_V2Wp8Reqor3NE6APuzomUfsSBGdk,5498
|
243
243
|
omlish/docker/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
244
244
|
omlish/docker/all.py,sha256=xXRgJgLGPwAtr7bDMJ_Dp9jTfOwfGvohNhc6LsoELJc,514
|
@@ -415,7 +415,7 @@ omlish/lang/generators.py,sha256=5LX17j-Ej3QXhwBgZvRTm_dq3n9veC4IOUcVmvSu2vU,524
|
|
415
415
|
omlish/lang/imports.py,sha256=Gdl6xCF89xiMOE1yDmdvKWamLq8HX-XPianO58Jdpmw,9218
|
416
416
|
omlish/lang/iterables.py,sha256=HOjcxOwyI5bBApDLsxRAGGhTTmw7fdZl2kEckxRVl-0,1994
|
417
417
|
omlish/lang/maybes.py,sha256=dAgrUoAhCgyrHRqa73CkaGnpXwGc-o9n-NIThrNXnbU,3416
|
418
|
-
omlish/lang/objects.py,sha256=
|
418
|
+
omlish/lang/objects.py,sha256=ih3z47DysrW11Vf3vF8rdALsnhK19Afp6wTU8AHAPWM,4982
|
419
419
|
omlish/lang/params.py,sha256=QmNVBfJsfxjDG5ilDPgHV7sK4UwRztkSQdLTo0umb8I,6648
|
420
420
|
omlish/lang/resolving.py,sha256=OuN2mDTPNyBUbcrswtvFKtj4xgH4H4WglgqSKv3MTy0,1606
|
421
421
|
omlish/lang/resources.py,sha256=WKkAddC3ctMK1bvGw-elGe8ZxAj2IaUTKVSu2nfgHTo,2839
|
@@ -532,7 +532,7 @@ omlish/os/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
|
532
532
|
omlish/os/atomics.py,sha256=KhWNeh4mzU3M-TF0v8uR6hUqMfZJW42MeyIK9Jl6R0k,5246
|
533
533
|
omlish/os/deathsig.py,sha256=hk9Yq2kyDdI-cI7OQH7mOfpRbOKzY_TfPKEqgrjVYbA,641
|
534
534
|
omlish/os/fcntl.py,sha256=riQf9iEEEIC28lJp8ud06MU56w2XJHJ9nBFtck_hdhc,1501
|
535
|
-
omlish/os/filemodes.py,sha256=
|
535
|
+
omlish/os/filemodes.py,sha256=ZIk6XpPw2oyoytSsuVmBXQ6fvMDHzctNEDIA2uHTBC4,4575
|
536
536
|
omlish/os/files.py,sha256=WJ_42vsZIZukQURN3TTccp-n74ZNhbux_ps3TLbHj18,1106
|
537
537
|
omlish/os/forkhooks.py,sha256=yjodOvs90ClXskv5oBIJbHn0Y7dzajLmZmOpRMKbyxM,5656
|
538
538
|
omlish/os/journald.py,sha256=2nI8Res1poXkbLc31--MPUlzYMESnCcPUkIxDOCjZW0,3903
|
@@ -670,7 +670,7 @@ omlish/sql/api/errors.py,sha256=YtC2gz5DqRTT3uCJniUOufVH1GEnFIc5ElkYLK3BHwM,230
|
|
670
670
|
omlish/sql/api/funcs.py,sha256=-H6V-o9JPSHFXsxdHtutB4mP2LwJfCzlLbRrPHunmB4,990
|
671
671
|
omlish/sql/api/queries.py,sha256=IgB8_sDe40-mKE-ByTmZ4GVOCdLLJDzJGJCevMd8R5s,1207
|
672
672
|
omlish/sql/api/rows.py,sha256=MEK9LNYEe8vLEJXQJD63MpnSOiE22cawJL-dUWQD6sU,1246
|
673
|
-
omlish/sql/parsing/Minisql.g4,sha256=
|
673
|
+
omlish/sql/parsing/Minisql.g4,sha256=Jw8xT-8UI6ySHAgStyCg5QX9NTCinvTenLJReWiZIJU,4578
|
674
674
|
omlish/sql/parsing/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
675
675
|
omlish/sql/parsing/parsing.py,sha256=hYIZdEKik4q22aEHHOX6lCpMa1jvgUk2aC1Llx10NMI,3343
|
676
676
|
omlish/sql/parsing/_antlr/MinisqlLexer.py,sha256=f4WzRrwqM7tW6sNn01035hRWPZL7xtuvba9EMR06j2E,17611
|
@@ -708,7 +708,7 @@ omlish/subprocesses/base.py,sha256=W6El-PUKKF9KLAks5LB6kzqs_n3FfkblJ-JOv6NFQbY,6
|
|
708
708
|
omlish/subprocesses/run.py,sha256=3jwSnQJvFMDMHmJvtAkrrK5D-i7_8cw12vX84EWTuJo,3668
|
709
709
|
omlish/subprocesses/sync.py,sha256=HKmKM99_Y7tkJRg_n5onXrw41IZt5M5fqU0281LY-mo,3671
|
710
710
|
omlish/subprocesses/utils.py,sha256=MJb6hvKhZceTmBeFVqlc5oM7rDxWkUzSzK9nKvbIvM8,396
|
711
|
-
omlish/subprocesses/wrap.py,sha256=
|
711
|
+
omlish/subprocesses/wrap.py,sha256=HMvCZrO2H227oGNN03KjB3FI-M5bAICqp19W8oG2f5M,763
|
712
712
|
omlish/term/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
713
713
|
omlish/term/codes.py,sha256=d8oii_9UJDm_TAowXYH3XDckX0XNBf2kiEPJKkVMMow,6149
|
714
714
|
omlish/term/progressbar.py,sha256=TiwdmPSMa5jQj35i1NQURTWQGy4eWUNx_XiPM38JtvQ,3184
|
@@ -746,16 +746,19 @@ omlish/testing/pytest/plugins/asyncs/backends/trio.py,sha256=xty9TR7-Kk6n0cdOqEr
|
|
746
746
|
omlish/testing/pytest/plugins/asyncs/backends/trio_asyncio.py,sha256=VcGVwf4V-1ZFK_70FrFS9b11EU1dOy1ozhhIDXGNSEo,3169
|
747
747
|
omlish/text/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
748
748
|
omlish/text/asdl.py,sha256=AS3irh-sag5pqyH3beJif78PjCbOaFso1NeKq-HXuTs,16867
|
749
|
+
omlish/text/decoding.py,sha256=sQWGckWzRslRHYKpj1SBeoo6AVqXm5HFlWFRARN1QpM,1286
|
749
750
|
omlish/text/delimit.py,sha256=Y0ID9Y9nfgQu3tYCiMS3hLa-ugA2cc-29PV4fF1343g,4927
|
751
|
+
omlish/text/filecache.py,sha256=ls08QSqBlhVXvjDwJpUXiP-U9HLyCstGAxtBOuWJmVY,5414
|
750
752
|
omlish/text/glyphsplit.py,sha256=kqqjglRdxGo0czYZxOz9Vi8aBmVsCOq8h6lPwRA5xe0,3803
|
751
753
|
omlish/text/indent.py,sha256=YjtJEBYWuk8--b9JU_T6q4yxV85_TR7VEVr5ViRCFwk,1336
|
754
|
+
omlish/text/linecache.py,sha256=hRYlEhD63ZfA6_ZOTkQIcnON-3W56QMAhcG3vEJqj9M,1858
|
752
755
|
omlish/text/mangle.py,sha256=kfzFLfvepH-chl1P89_mdc5vC4FSqyPA2aVtgzuB8IY,1133
|
753
756
|
omlish/text/minja.py,sha256=jZC-fp3Xuhx48ppqsf2Sf1pHbC0t8XBB7UpUUoOk2Qw,5751
|
754
|
-
omlish/text/parts.py,sha256=
|
757
|
+
omlish/text/parts.py,sha256=Q9NvoyEGQKIWgiPD4D_Qc66cWAuyEKE033dT9m7c3Wk,6662
|
755
758
|
omlish/text/random.py,sha256=jNWpqiaKjKyTdMXC-pWAsSC10AAP-cmRRPVhm59ZWLk,194
|
756
|
-
omlish-0.0.0.
|
757
|
-
omlish-0.0.0.
|
758
|
-
omlish-0.0.0.
|
759
|
-
omlish-0.0.0.
|
760
|
-
omlish-0.0.0.
|
761
|
-
omlish-0.0.0.
|
759
|
+
omlish-0.0.0.dev255.dist-info/LICENSE,sha256=B_hVtavaA8zCYDW99DYdcpDLKz1n3BBRjZrcbv8uG8c,1451
|
760
|
+
omlish-0.0.0.dev255.dist-info/METADATA,sha256=OYcI4_aud4-SKLb5y1OiYRjgrj-QkyxBctM6FPPe9aQ,4176
|
761
|
+
omlish-0.0.0.dev255.dist-info/WHEEL,sha256=beeZ86-EfXScwlR_HKu4SllMC9wUEj_8Z_4FJ3egI2w,91
|
762
|
+
omlish-0.0.0.dev255.dist-info/entry_points.txt,sha256=Lt84WvRZJskWCAS7xnQGZIeVWksprtUHj0llrvVmod8,35
|
763
|
+
omlish-0.0.0.dev255.dist-info/top_level.txt,sha256=pePsKdLu7DvtUiecdYXJ78iO80uDNmBlqe-8hOzOmfs,7
|
764
|
+
omlish-0.0.0.dev255.dist-info/RECORD,,
|
File without changes
|
File without changes
|
File without changes
|