omlish 0.0.0.dev72__py3-none-any.whl → 0.0.0.dev74__py3-none-any.whl

Sign up to get free protection for your applications and to get access to all the features.
omlish/__about__.py CHANGED
@@ -1,5 +1,5 @@
1
- __version__ = '0.0.0.dev72'
2
- __revision__ = '4448e03bbb77cb149e46eeefb0e9e61faed2a494'
1
+ __version__ = '0.0.0.dev74'
2
+ __revision__ = '002d07c8f9579e6b60584c9b2c6c05cd6b7219af'
3
3
 
4
4
 
5
5
  #
omlish/check.py CHANGED
@@ -96,7 +96,7 @@ def _default_exception_factory(exc_cls: type[Exception], *args, **kwargs) -> Exc
96
96
  _EXCEPTION_FACTORY = _default_exception_factory
97
97
 
98
98
 
99
- class _Args:
99
+ class _ArgsKwargs:
100
100
  def __init__(self, *args, **kwargs):
101
101
  self.args = args
102
102
  self.kwargs = kwargs
@@ -106,7 +106,7 @@ def _raise(
106
106
  exception_type: type[Exception],
107
107
  default_message: str,
108
108
  message: Message,
109
- ak: _Args = _Args(),
109
+ ak: _ArgsKwargs = _ArgsKwargs(),
110
110
  *,
111
111
  render_fmt: str | None = None,
112
112
  ) -> ta.NoReturn:
@@ -159,7 +159,7 @@ def isinstance(v: ta.Any, spec: type[T] | tuple, msg: Message = None) -> T: # n
159
159
  TypeError,
160
160
  'Must be instance',
161
161
  msg,
162
- _Args(v, spec),
162
+ _ArgsKwargs(v, spec),
163
163
  render_fmt='not isinstance(%s, %s)',
164
164
  )
165
165
 
@@ -179,7 +179,7 @@ def cast(v: ta.Any, cls: type[T], msg: Message = None) -> T: # noqa
179
179
  TypeError,
180
180
  'Must be instance',
181
181
  msg,
182
- _Args(v, cls),
182
+ _ArgsKwargs(v, cls),
183
183
  )
184
184
 
185
185
  return v
@@ -198,7 +198,7 @@ def not_isinstance(v: T, spec: ta.Any, msg: Message = None) -> T: # noqa
198
198
  TypeError,
199
199
  'Must not be instance',
200
200
  msg,
201
- _Args(v, spec),
201
+ _ArgsKwargs(v, spec),
202
202
  render_fmt='isinstance(%s, %s)',
203
203
  )
204
204
 
@@ -221,7 +221,7 @@ def issubclass(v: type[T], spec: ta.Any, msg: Message = None) -> type[T]: # noq
221
221
  TypeError,
222
222
  'Must be subclass',
223
223
  msg,
224
- _Args(v, spec),
224
+ _ArgsKwargs(v, spec),
225
225
  render_fmt='not issubclass(%s, %s)',
226
226
  )
227
227
 
@@ -234,7 +234,7 @@ def not_issubclass(v: type[T], spec: ta.Any, msg: Message = None) -> type[T]: #
234
234
  TypeError,
235
235
  'Must not be subclass',
236
236
  msg,
237
- _Args(v, spec),
237
+ _ArgsKwargs(v, spec),
238
238
  render_fmt='issubclass(%s, %s)',
239
239
  )
240
240
 
@@ -250,7 +250,7 @@ def in_(v: T, c: ta.Container[T], msg: Message = None) -> T:
250
250
  ValueError,
251
251
  'Must be in',
252
252
  msg,
253
- _Args(v, c),
253
+ _ArgsKwargs(v, c),
254
254
  render_fmt='%s not in %s',
255
255
  )
256
256
 
@@ -263,7 +263,7 @@ def not_in(v: T, c: ta.Container[T], msg: Message = None) -> T:
263
263
  ValueError,
264
264
  'Must not be in',
265
265
  msg,
266
- _Args(v, c),
266
+ _ArgsKwargs(v, c),
267
267
  render_fmt='%s in %s',
268
268
  )
269
269
 
@@ -276,7 +276,7 @@ def empty(v: SizedT, msg: Message = None) -> SizedT:
276
276
  ValueError,
277
277
  'Must be empty',
278
278
  msg,
279
- _Args(v),
279
+ _ArgsKwargs(v),
280
280
  render_fmt='%s',
281
281
  )
282
282
 
@@ -294,7 +294,7 @@ def iterempty(v: ta.Iterable[T], msg: Message = None) -> ta.Iterable[T]:
294
294
  ValueError,
295
295
  'Must be empty',
296
296
  msg,
297
- _Args(v),
297
+ _ArgsKwargs(v),
298
298
  render_fmt='%s',
299
299
  )
300
300
 
@@ -307,7 +307,7 @@ def not_empty(v: SizedT, msg: Message = None) -> SizedT:
307
307
  ValueError,
308
308
  'Must not be empty',
309
309
  msg,
310
- _Args(v),
310
+ _ArgsKwargs(v),
311
311
  render_fmt='%s',
312
312
  )
313
313
 
@@ -321,7 +321,7 @@ def unique(it: ta.Iterable[T], msg: Message = None) -> ta.Iterable[T]:
321
321
  ValueError,
322
322
  'Must be unique',
323
323
  msg,
324
- _Args(it, dupes),
324
+ _ArgsKwargs(it, dupes),
325
325
  )
326
326
 
327
327
  return it
@@ -335,7 +335,7 @@ def single(obj: ta.Iterable[T], message: Message = None) -> T:
335
335
  ValueError,
336
336
  'Must be single',
337
337
  message,
338
- _Args(obj),
338
+ _ArgsKwargs(obj),
339
339
  render_fmt='%s',
340
340
  )
341
341
 
@@ -358,7 +358,7 @@ def opt_single(obj: ta.Iterable[T], message: Message = None) -> T | None:
358
358
  ValueError,
359
359
  'Must be empty or single',
360
360
  message,
361
- _Args(obj),
361
+ _ArgsKwargs(obj),
362
362
  render_fmt='%s',
363
363
  )
364
364
 
@@ -372,7 +372,7 @@ def none(v: ta.Any, msg: Message = None) -> None:
372
372
  ValueError,
373
373
  'Must be None',
374
374
  msg,
375
- _Args(v),
375
+ _ArgsKwargs(v),
376
376
  render_fmt='%s',
377
377
  )
378
378
 
@@ -383,7 +383,7 @@ def not_none(v: T | None, msg: Message = None) -> T:
383
383
  ValueError,
384
384
  'Must not be None',
385
385
  msg,
386
- _Args(v),
386
+ _ArgsKwargs(v),
387
387
  render_fmt='%s',
388
388
  )
389
389
 
@@ -399,7 +399,7 @@ def equal(v: T, o: ta.Any, msg: Message = None) -> T:
399
399
  ValueError,
400
400
  'Must be equal',
401
401
  msg,
402
- _Args(v, o),
402
+ _ArgsKwargs(v, o),
403
403
  render_fmt='%s != %s',
404
404
  )
405
405
 
@@ -412,7 +412,7 @@ def is_(v: T, o: ta.Any, msg: Message = None) -> T:
412
412
  ValueError,
413
413
  'Must be the same',
414
414
  msg,
415
- _Args(v, o),
415
+ _ArgsKwargs(v, o),
416
416
  render_fmt='%s is not %s',
417
417
  )
418
418
 
@@ -425,7 +425,7 @@ def is_not(v: T, o: ta.Any, msg: Message = None) -> T:
425
425
  ValueError,
426
426
  'Must not be the same',
427
427
  msg,
428
- _Args(v, o),
428
+ _ArgsKwargs(v, o),
429
429
  render_fmt='%s is %s',
430
430
  )
431
431
 
@@ -438,7 +438,7 @@ def callable(v: T, msg: Message = None) -> T: # noqa
438
438
  TypeError,
439
439
  'Must be callable',
440
440
  msg,
441
- _Args(v),
441
+ _ArgsKwargs(v),
442
442
  render_fmt='%s',
443
443
  )
444
444
 
@@ -451,7 +451,7 @@ def non_empty_str(v: str | None, msg: Message = None) -> str:
451
451
  ValueError,
452
452
  'Must be non-empty str',
453
453
  msg,
454
- _Args(v),
454
+ _ArgsKwargs(v),
455
455
  render_fmt='%s',
456
456
  )
457
457
 
@@ -464,7 +464,7 @@ def replacing(expected: ta.Any, old: ta.Any, new: T, msg: Message = None) -> T:
464
464
  ValueError,
465
465
  'Must be replacing',
466
466
  msg,
467
- _Args(expected, old, new),
467
+ _ArgsKwargs(expected, old, new),
468
468
  render_fmt='%s -> %s -> %s',
469
469
  )
470
470
 
@@ -477,7 +477,7 @@ def replacing_none(old: ta.Any, new: T, msg: Message = None) -> T:
477
477
  ValueError,
478
478
  'Must be replacing None',
479
479
  msg,
480
- _Args(old, new),
480
+ _ArgsKwargs(old, new),
481
481
  render_fmt='%s -> %s',
482
482
  )
483
483
 
@@ -493,7 +493,7 @@ def arg(v: bool, msg: Message = None) -> None:
493
493
  RuntimeError,
494
494
  'Argument condition not met',
495
495
  msg,
496
- _Args(v),
496
+ _ArgsKwargs(v),
497
497
  render_fmt='%s',
498
498
  )
499
499
 
@@ -504,6 +504,6 @@ def state(v: bool, msg: Message = None) -> None:
504
504
  RuntimeError,
505
505
  'State condition not met',
506
506
  msg,
507
- _Args(v),
507
+ _ArgsKwargs(v),
508
508
  render_fmt='%s',
509
509
  )
@@ -1,3 +1,7 @@
1
+ """
2
+ TODO:
3
+ - point validate / check exceptions to lambdas
4
+ """
1
5
  import dataclasses as dc
2
6
  import types
3
7
  import typing as ta
@@ -108,7 +108,11 @@ class DataMeta(abc.ABCMeta):
108
108
 
109
109
 
110
110
  # @ta.dataclass_transform(field_specifiers=(field,)) # FIXME: ctor
111
- class Data(metaclass=DataMeta):
111
+ class Data(
112
+ eq=False,
113
+ order=False,
114
+ metaclass=DataMeta,
115
+ ):
112
116
  def __init__(self, *args, **kwargs):
113
117
  super().__init__(*args, **kwargs)
114
118
 
@@ -125,6 +129,8 @@ class Data(metaclass=DataMeta):
125
129
  class Frozen(
126
130
  Data,
127
131
  frozen=True,
132
+ eq=False,
133
+ order=False,
128
134
  confer=frozenset([
129
135
  'frozen',
130
136
  'cache_hash',
omlish/http/clients.py CHANGED
@@ -133,7 +133,7 @@ class UrllibHttpClient(HttpClient):
133
133
 
134
134
  # urllib headers are dumb dicts [1], and keys *must* be strings or it will automatically add problematic default
135
135
  # headers because it doesn't see string keys in its header dict [2]. frustratingly it has no problem accepting
136
- # bytes keys though [3].
136
+ # bytes values though [3].
137
137
  # [1]: https://github.com/python/cpython/blob/232b303e4ca47892f544294bf42e31dc34f0ec72/Lib/urllib/request.py#L319-L325 # noqa
138
138
  # [2]: https://github.com/python/cpython/blob/232b303e4ca47892f544294bf42e31dc34f0ec72/Lib/urllib/request.py#L1276-L1279 # noqa
139
139
  # [3]: https://github.com/python/cpython/blob/232b303e4ca47892f544294bf42e31dc34f0ec72/Lib/http/client.py#L1300-L1301 # noqa
omlish/http/headers.py CHANGED
@@ -12,13 +12,14 @@ CanHttpHeaders: ta.TypeAlias = ta.Union[
12
12
 
13
13
  ta.Mapping[str, str],
14
14
  ta.Mapping[str, ta.Sequence[str]],
15
+ ta.Mapping[str, str | ta.Sequence[str]],
15
16
 
16
17
  ta.Mapping[bytes, bytes],
17
18
  ta.Mapping[bytes, ta.Sequence[bytes]],
19
+ ta.Mapping[bytes, bytes | ta.Sequence[bytes]],
18
20
 
19
21
  ta.Mapping[StrOrBytes, StrOrBytes],
20
22
  ta.Mapping[StrOrBytes, ta.Sequence[StrOrBytes]],
21
-
22
23
  ta.Mapping[StrOrBytes, StrOrBytes | ta.Sequence[StrOrBytes]],
23
24
 
24
25
  ta.Sequence[tuple[str, str]],
@@ -153,11 +154,11 @@ class HttpHeaders:
153
154
  ...
154
155
 
155
156
  @ta.overload
156
- def __getitem__(self, item: int) -> StrOrBytes:
157
+ def __getitem__(self, item: int) -> tuple[StrOrBytes, StrOrBytes]:
157
158
  ...
158
159
 
159
160
  @ta.overload
160
- def __getitem__(self, item: slice) -> ta.Sequence[StrOrBytes]:
161
+ def __getitem__(self, item: slice) -> ta.Sequence[tuple[StrOrBytes, StrOrBytes]]:
161
162
  ...
162
163
 
163
164
  def __getitem__(self, item):
omlish/marshal/enums.py CHANGED
@@ -28,7 +28,7 @@ class EnumMarshalerFactory(MarshalerFactory):
28
28
  def fn(self, ctx: MarshalContext, rty: rfl.Type) -> Marshaler:
29
29
  ty = check.isinstance(rty, type)
30
30
  check.state(issubclass(ty, enum.Enum))
31
- return EnumMarshaler(ty)
31
+ return EnumMarshaler(ty) # noqa
32
32
 
33
33
 
34
34
  @dc.dataclass(frozen=True)
@@ -11,6 +11,10 @@ from .binary import ( # noqa
11
11
  BinaryOps,
12
12
  )
13
13
 
14
+ from .building import ( # noqa
15
+ StdBuilder,
16
+ )
17
+
14
18
  from .exprs import ( # noqa
15
19
  CanExpr,
16
20
  CanLiteral,
@@ -18,6 +22,7 @@ from .exprs import ( # noqa
18
22
  ExprBuilder,
19
23
  Literal,
20
24
  NameExpr,
25
+ Param,
21
26
  )
22
27
 
23
28
  from .idents import ( # noqa
@@ -29,8 +34,7 @@ from .idents import ( # noqa
29
34
  from .multi import ( # noqa
30
35
  Multi,
31
36
  MultiBuilder,
32
- MultiOp,
33
- MultiOps,
37
+ MultiKind,
34
38
  )
35
39
 
36
40
  from .names import ( # noqa
@@ -39,6 +43,10 @@ from .names import ( # noqa
39
43
  NameBuilder,
40
44
  )
41
45
 
46
+ from .ops import ( # noqa
47
+ OpKind,
48
+ )
49
+
42
50
  from .relations import ( # noqa
43
51
  CanRelation,
44
52
  CanTable,
@@ -47,6 +55,12 @@ from .relations import ( # noqa
47
55
  Table,
48
56
  )
49
57
 
58
+ from .rendering import ( # noqa
59
+ Renderer,
60
+ StdRenderer,
61
+ render,
62
+ )
63
+
50
64
  from .selects import ( # noqa
51
65
  CanRelation,
52
66
  Select,
@@ -54,10 +68,6 @@ from .selects import ( # noqa
54
68
  SelectItem,
55
69
  )
56
70
 
57
- from .std import ( # noqa
58
- StdBuilder,
59
- )
60
-
61
71
  from .stmts import ( # noqa
62
72
  CanExpr,
63
73
  ExprStmt,
@@ -77,3 +87,11 @@ from .unary import ( # noqa
77
87
 
78
88
 
79
89
  Q = StdBuilder()
90
+
91
+
92
+ ##
93
+
94
+
95
+ from ...lang.imports import _register_conditional_import # noqa
96
+
97
+ _register_conditional_import('...marshal', '.marshal', __package__)
@@ -1,24 +1,41 @@
1
+ """
2
+ TODO:
3
+ - in_
4
+ - like
5
+ - no this is a dedicated node, escape / negation in grammar
6
+ """
1
7
  from ... import check
8
+ from ... import dataclasses as dc
2
9
  from ... import lang
3
- from .base import Node
4
10
  from .exprs import CanExpr
5
11
  from .exprs import Expr
6
12
  from .exprs import ExprBuilder
13
+ from .ops import OpKind
7
14
 
8
15
 
9
16
  ##
10
17
 
11
18
 
12
- class BinaryOp(Node, lang.Final):
19
+ class BinaryOp(dc.Frozen, lang.Final, eq=False):
13
20
  name: str
21
+ kind: OpKind
14
22
 
15
23
 
16
24
  class BinaryOps(lang.Namespace):
17
- ADD = BinaryOp('add')
18
- SUB = BinaryOp('sub')
25
+ EQ = BinaryOp('eq', OpKind.CMP)
26
+ NE = BinaryOp('ne', OpKind.CMP)
27
+ LT = BinaryOp('lt', OpKind.CMP)
28
+ LE = BinaryOp('le', OpKind.CMP)
29
+ GT = BinaryOp('gt', OpKind.CMP)
30
+ GE = BinaryOp('ge', OpKind.CMP)
19
31
 
20
- EQ = BinaryOp('eq')
21
- NE = BinaryOp('ne')
32
+ ADD = BinaryOp('add', OpKind.ARITH)
33
+ SUB = BinaryOp('sub', OpKind.ARITH)
34
+ MUL = BinaryOp('mul', OpKind.ARITH)
35
+ DIV = BinaryOp('div', OpKind.ARITH)
36
+ MOD = BinaryOp('mod', OpKind.ARITH)
37
+
38
+ CONCAT = BinaryOp('concat', OpKind.STR)
22
39
 
23
40
 
24
41
  class Binary(Expr, lang.Final):
@@ -35,14 +52,44 @@ class BinaryBuilder(ExprBuilder):
35
52
  l = Binary(op, l, self.expr(r))
36
53
  return l
37
54
 
55
+ #
56
+
57
+ def eq(self, *es: CanExpr) -> Expr:
58
+ return self.binary(BinaryOps.EQ, *es)
59
+
60
+ def ne(self, *es: CanExpr) -> Expr:
61
+ return self.binary(BinaryOps.NE, *es)
62
+
63
+ def lt(self, *es: CanExpr) -> Expr:
64
+ return self.binary(BinaryOps.LT, *es)
65
+
66
+ def le(self, *es: CanExpr) -> Expr:
67
+ return self.binary(BinaryOps.LE, *es)
68
+
69
+ def gt(self, *es: CanExpr) -> Expr:
70
+ return self.binary(BinaryOps.GT, *es)
71
+
72
+ def ge(self, *es: CanExpr) -> Expr:
73
+ return self.binary(BinaryOps.GE, *es)
74
+
75
+ #
76
+
38
77
  def add(self, *es: CanExpr) -> Expr:
39
78
  return self.binary(BinaryOps.ADD, *es)
40
79
 
41
80
  def sub(self, *es: CanExpr) -> Expr:
42
81
  return self.binary(BinaryOps.SUB, *es)
43
82
 
44
- def eq(self, *es: CanExpr) -> Expr:
45
- return self.binary(BinaryOps.EQ, *es)
83
+ def mul(self, *es: CanExpr) -> Expr:
84
+ return self.binary(BinaryOps.MUL, *es)
46
85
 
47
- def ne(self, *es: CanExpr) -> Expr:
48
- return self.binary(BinaryOps.NE, *es)
86
+ def div(self, *es: CanExpr) -> Expr:
87
+ return self.binary(BinaryOps.DIV, *es)
88
+
89
+ def mod(self, *es: CanExpr) -> Expr:
90
+ return self.binary(BinaryOps.MOD, *es)
91
+
92
+ #
93
+
94
+ def concat(self, *es: CanExpr) -> Expr:
95
+ return self.binary(BinaryOps.CONCAT, *es)
@@ -1,3 +1,8 @@
1
+ """
2
+ TODO:
3
+ - case
4
+ - cast / ::
5
+ """
1
6
  import typing as ta
2
7
 
3
8
  from ... import lang
@@ -24,6 +29,24 @@ class NameExpr(Expr, lang.Final):
24
29
  n: Name
25
30
 
26
31
 
32
+ class Param(Expr, lang.Final):
33
+ n: str | None = None
34
+
35
+ def __repr__(self) -> str:
36
+ if self.n is not None:
37
+ return f'{self.__class__.__name__}({self.n!r})'
38
+ else:
39
+ return f'{self.__class__.__name__}(@{hex(id(self))[2:]})'
40
+
41
+ def __eq__(self, other):
42
+ if not isinstance(other, Param):
43
+ return False
44
+ if self.n is None and other.n is None:
45
+ return self is other
46
+ else:
47
+ return self.n == other.n
48
+
49
+
27
50
  CanLiteral: ta.TypeAlias = Literal | Value
28
51
  CanExpr: ta.TypeAlias = Expr | CanName | CanLiteral
29
52
 
@@ -41,6 +64,8 @@ class ExprBuilder(NameBuilder):
41
64
  def l(self, o: CanLiteral) -> Literal: # noqa
42
65
  return self.literal(o)
43
66
 
67
+ #
68
+
44
69
  def expr(self, o: CanExpr) -> Expr:
45
70
  if isinstance(o, Expr):
46
71
  return o
@@ -52,3 +77,12 @@ class ExprBuilder(NameBuilder):
52
77
  @ta.final
53
78
  def e(self, o: CanExpr) -> Expr:
54
79
  return self.expr(o)
80
+
81
+ #
82
+
83
+ def param(self, n: str | None = None) -> Param:
84
+ return Param(n)
85
+
86
+ @ta.final
87
+ def p(self, n: str | None = None) -> Param:
88
+ return self.param(n)
@@ -0,0 +1,74 @@
1
+ import enum
2
+ import typing as ta
3
+
4
+ from ... import cached
5
+ from ... import check
6
+ from ... import collections as col
7
+ from ... import dataclasses as dc
8
+ from ... import lang
9
+ from ... import marshal as msh
10
+ from .base import Node
11
+ from .binary import BinaryOp
12
+ from .binary import BinaryOps
13
+ from .exprs import Expr
14
+ from .multi import MultiKind
15
+ from .relations import Relation
16
+ from .stmts import Stmt
17
+ from .unary import UnaryOp
18
+ from .unary import UnaryOps
19
+
20
+
21
+ @dc.dataclass(frozen=True)
22
+ class OpMarshalerUnmarshaler(msh.Marshaler, msh.Unmarshaler):
23
+ ty: type
24
+ ns: type[lang.Namespace]
25
+
26
+ @cached.property
27
+ @dc.init
28
+ def by_name(self) -> ta.Mapping[str, ta.Any]:
29
+ return col.make_map(((o.name, o) for _, o in self.ns), strict=True)
30
+
31
+ def marshal(self, ctx: msh.MarshalContext, o: ta.Any) -> msh.Value:
32
+ return check.isinstance(o, self.ty).name # type: ignore # noqa
33
+
34
+ def unmarshal(self, ctx: msh.UnmarshalContext, v: msh.Value) -> ta.Any:
35
+ return self.by_name[check.isinstance(v, str)]
36
+
37
+
38
+ @dc.dataclass(frozen=True)
39
+ class LowerEnumMarshaler(msh.Marshaler, msh.Unmarshaler):
40
+ ty: type[enum.Enum]
41
+
42
+ @cached.property
43
+ @dc.init
44
+ def by_name(self) -> ta.Mapping[str, ta.Any]:
45
+ return col.make_map(((o.name.lower(), o) for o in self.ty), strict=True)
46
+
47
+ def marshal(self, ctx: msh.MarshalContext, o: ta.Any) -> msh.Value:
48
+ return o.name.lower()
49
+
50
+ def unmarshal(self, ctx: msh.UnmarshalContext, v: msh.Value) -> ta.Any:
51
+ return self.by_name[check.isinstance(v, str).lower()]
52
+
53
+
54
+ @lang.static_init
55
+ def _install_standard_marshalling() -> None:
56
+ for ty, ns in [
57
+ (BinaryOp, BinaryOps),
58
+ (UnaryOp, UnaryOps),
59
+ ]:
60
+ msh.STANDARD_MARSHALER_FACTORIES[0:0] = [msh.TypeMapMarshalerFactory({ty: OpMarshalerUnmarshaler(ty, ns)})]
61
+ msh.STANDARD_UNMARSHALER_FACTORIES[0:0] = [msh.TypeMapUnmarshalerFactory({ty: OpMarshalerUnmarshaler(ty, ns)})]
62
+
63
+ msh.STANDARD_MARSHALER_FACTORIES[0:0] = [msh.TypeMapMarshalerFactory({MultiKind: LowerEnumMarshaler(MultiKind)})]
64
+ msh.STANDARD_UNMARSHALER_FACTORIES[0:0] = [msh.TypeMapUnmarshalerFactory({MultiKind: LowerEnumMarshaler(MultiKind)})] # noqa
65
+
66
+ for cls in [
67
+ Expr,
68
+ Node,
69
+ Relation,
70
+ Stmt,
71
+ ]:
72
+ p = msh.polymorphism_from_subclasses(cls, naming=msh.Naming.SNAKE)
73
+ msh.STANDARD_MARSHALER_FACTORIES[0:0] = [msh.PolymorphismMarshalerFactory(p)]
74
+ msh.STANDARD_UNMARSHALER_FACTORIES[0:0] = [msh.PolymorphismUnmarshalerFactory(p)]
@@ -1,9 +1,9 @@
1
+ import enum
1
2
  import typing as ta
2
3
 
3
4
  from ... import check
4
5
  from ... import dataclasses as dc
5
6
  from ... import lang
6
- from .base import Node
7
7
  from .exprs import CanExpr
8
8
  from .exprs import Expr
9
9
  from .exprs import ExprBuilder
@@ -12,30 +12,26 @@ from .exprs import ExprBuilder
12
12
  ##
13
13
 
14
14
 
15
- class MultiOp(Node, lang.Final):
16
- name: str
17
-
18
-
19
- class MultiOps(lang.Namespace):
20
- AND = MultiOp('and')
21
- OR = MultiOp('or')
15
+ class MultiKind(enum.Enum):
16
+ AND = enum.auto()
17
+ OR = enum.auto()
22
18
 
23
19
 
24
20
  class Multi(Expr, lang.Final):
25
- op: MultiOp
26
- es: ta.Sequence[Expr] = dc.xfield(coerce=tuple)
21
+ k: MultiKind
22
+ es: ta.Sequence[Expr] = dc.xfield(coerce=tuple, validate=lambda es: len(es) > 1)
27
23
 
28
24
 
29
25
  class MultiBuilder(ExprBuilder):
30
- def multi(self, op: MultiOp, *es: CanExpr) -> Expr:
26
+ def multi(self, k: MultiKind, *es: CanExpr) -> Expr:
31
27
  check.not_empty(es)
32
28
  if len(es) == 1:
33
29
  return self.expr(es[0])
34
30
  else:
35
- return Multi(op, [self.expr(e) for e in es])
31
+ return Multi(k, [self.expr(e) for e in es])
36
32
 
37
33
  def and_(self, *es: CanExpr) -> Expr:
38
- return self.multi(MultiOps.AND, *es)
34
+ return self.multi(MultiKind.AND, *es)
39
35
 
40
36
  def or_(self, *es: CanExpr) -> Expr:
41
- return self.multi(MultiOps.OR, *es)
37
+ return self.multi(MultiKind.OR, *es)
@@ -0,0 +1,8 @@
1
+ import enum
2
+
3
+
4
+ class OpKind(enum.Enum):
5
+ CMP = enum.auto()
6
+ ARITH = enum.auto()
7
+ BIT = enum.auto()
8
+ STR = enum.auto()
@@ -1,3 +1,8 @@
1
+ """
2
+ TODO:
3
+ - join
4
+ - subquery
5
+ """
1
6
  import typing as ta
2
7
 
3
8
  from ... import dataclasses as dc
@@ -0,0 +1,180 @@
1
+ """
2
+ TODO:
3
+ - minimal parens
4
+ - text.parts
5
+ - QuoteStyle
6
+ - ParamStyle
7
+
8
+ ==
9
+
10
+ def needs_parens(self, e: Expr) -> bool:
11
+ if isinstance(e, (Literal, Ident, Name)):
12
+ return True
13
+ elif isinstance(e, Expr):
14
+ return False
15
+ else:
16
+ raise TypeError(e)
17
+ """
18
+ import io
19
+ import typing as ta
20
+
21
+ from ... import dispatch
22
+ from ... import lang
23
+ from .base import Node
24
+ from .binary import Binary
25
+ from .binary import BinaryOp
26
+ from .binary import BinaryOps
27
+ from .exprs import Literal
28
+ from .exprs import NameExpr
29
+ from .idents import Ident
30
+ from .multi import Multi
31
+ from .multi import MultiKind
32
+ from .names import Name
33
+ from .relations import Table
34
+ from .selects import Select
35
+ from .selects import SelectItem
36
+ from .unary import Unary
37
+ from .unary import UnaryOp
38
+ from .unary import UnaryOps
39
+
40
+
41
+ class Renderer(lang.Abstract):
42
+ def __init__(self, out: ta.TextIO) -> None:
43
+ super().__init__()
44
+ self._out = out
45
+
46
+ @dispatch.method
47
+ def render(self, o: ta.Any) -> None:
48
+ raise TypeError(o)
49
+
50
+ @classmethod
51
+ def render_str(cls, o: ta.Any, *args: ta.Any, **kwargs: ta.Any) -> str:
52
+ out = io.StringIO()
53
+ cls(out, *args, **kwargs).render(o)
54
+ return out.getvalue()
55
+
56
+
57
+ class StdRenderer(Renderer):
58
+ # binary
59
+
60
+ BINARY_OP_TO_STR: ta.ClassVar[ta.Mapping[BinaryOp, str]] = {
61
+ BinaryOps.EQ: '=',
62
+ BinaryOps.NE: '!=',
63
+ BinaryOps.LT: '<',
64
+ BinaryOps.LE: '<=',
65
+ BinaryOps.GT: '>',
66
+ BinaryOps.GE: '>=',
67
+
68
+ BinaryOps.ADD: '+',
69
+ BinaryOps.SUB: '-',
70
+ BinaryOps.MUL: '*',
71
+ BinaryOps.DIV: '/',
72
+ BinaryOps.MOD: '%',
73
+
74
+ BinaryOps.CONCAT: '||',
75
+ }
76
+
77
+ @Renderer.render.register
78
+ def render_binary(self, o: Binary) -> None:
79
+ self._out.write('(')
80
+ self.render(o.l)
81
+ self._out.write(f' {self.BINARY_OP_TO_STR[o.op]} ')
82
+ self.render(o.r)
83
+ self._out.write(')')
84
+
85
+ # exprs
86
+
87
+ @Renderer.render.register
88
+ def render_literal(self, o: Literal) -> None:
89
+ self._out.write(repr(o.v))
90
+
91
+ @Renderer.render.register
92
+ def render_name_expr(self, o: NameExpr) -> None:
93
+ self.render(o.n)
94
+
95
+ # idents
96
+
97
+ @Renderer.render.register
98
+ def render_ident(self, o: Ident) -> None:
99
+ self._out.write(f'"{o.s}"')
100
+
101
+ # multis
102
+
103
+ MULTI_KIND_TO_STR: ta.ClassVar[ta.Mapping[MultiKind, str]] = {
104
+ MultiKind.AND: 'and',
105
+ MultiKind.OR: 'or',
106
+
107
+ }
108
+
109
+ @Renderer.render.register
110
+ def render_multi(self, o: Multi) -> None:
111
+ d = f' {self.MULTI_KIND_TO_STR[o.k]} '
112
+ self._out.write('(')
113
+ for i, e in enumerate(o.es):
114
+ if i:
115
+ self._out.write(d)
116
+ self.render(e)
117
+ self._out.write(')')
118
+
119
+ # names
120
+
121
+ @Renderer.render.register
122
+ def render_name(self, o: Name) -> None:
123
+ for n, i in enumerate(o.ps):
124
+ if n:
125
+ self._out.write('.')
126
+ self.render(i)
127
+
128
+ # relations
129
+
130
+ @Renderer.render.register
131
+ def render_table(self, o: Table) -> None:
132
+ self.render(o.n)
133
+ if o.a is not None:
134
+ self._out.write(' as ')
135
+ self.render(o.a)
136
+
137
+ # selects
138
+
139
+ @Renderer.render.register
140
+ def render_select_item(self, o: SelectItem) -> None:
141
+ self.render(o.v)
142
+ if o.a is not None:
143
+ self._out.write(' as ')
144
+ self.render(o.a)
145
+
146
+ @Renderer.render.register
147
+ def render_select(self, o: Select) -> None:
148
+ self._out.write('select ')
149
+ for i, it in enumerate(o.items):
150
+ if i:
151
+ self._out.write(', ')
152
+ self.render(it)
153
+ if o.from_ is not None:
154
+ self._out.write(' from ')
155
+ self.render(o.from_)
156
+ if o.where:
157
+ self._out.write(' where ')
158
+ self.render(o.where)
159
+
160
+ # unary
161
+
162
+ UNARY_OP_TO_STR: ta.ClassVar[ta.Mapping[UnaryOp, tuple[str, str]]] = {
163
+ UnaryOps.NOT: ('not ', ''),
164
+ UnaryOps.IS_NULL: ('', ' is null'),
165
+ UnaryOps.IS_NOT_NULL: ('', ' is not null'),
166
+
167
+ UnaryOps.POS: ('+', ''),
168
+ UnaryOps.NEG: ('-', ''),
169
+ }
170
+
171
+ @Renderer.render.register
172
+ def render_unary(self, o: Unary) -> None:
173
+ pfx, sfx = self.UNARY_OP_TO_STR[o.op]
174
+ self._out.write(pfx)
175
+ self.render(o.v)
176
+ self._out.write(sfx)
177
+
178
+
179
+ def render(n: Node) -> str:
180
+ return StdRenderer.render_str(n)
@@ -22,9 +22,9 @@ class SelectItem(Node, lang.Final):
22
22
 
23
23
 
24
24
  class Select(Stmt, lang.Final):
25
- its: ta.Sequence[SelectItem] = dc.xfield(coerce=tuple)
26
- fr: Relation | None = dc.xfield(None, repr_fn=dc.opt_repr)
27
- wh: Expr | None = dc.xfield(None, repr_fn=dc.opt_repr)
25
+ items: ta.Sequence[SelectItem] = dc.xfield(coerce=tuple)
26
+ from_: Relation | None = dc.xfield(None, repr_fn=dc.opt_repr)
27
+ where: Expr | None = dc.xfield(None, repr_fn=dc.opt_repr)
28
28
 
29
29
 
30
30
  CanSelectItem: ta.TypeAlias = SelectItem | CanExpr
@@ -39,12 +39,12 @@ class SelectBuilder(ExprBuilder, RelationBuilder):
39
39
 
40
40
  def select(
41
41
  self,
42
- its: ta.Sequence[CanSelectItem],
43
- fr: CanRelation | None = None,
44
- wh: CanExpr | None = None,
42
+ items: ta.Sequence[CanSelectItem],
43
+ from_: CanRelation | None = None,
44
+ where: CanExpr | None = None,
45
45
  ) -> Select:
46
46
  return Select(
47
- [self.select_item(i) for i in its],
48
- fr=self.relation(fr) if fr is not None else None,
49
- wh=self.expr(wh) if wh is not None else None,
47
+ [self.select_item(i) for i in items],
48
+ from_=self.relation(from_) if from_ is not None else None,
49
+ where=self.expr(where) if where is not None else None,
50
50
  )
@@ -1,19 +1,26 @@
1
+ from ... import dataclasses as dc
1
2
  from ... import lang
2
- from .base import Node
3
3
  from .exprs import CanExpr
4
4
  from .exprs import Expr
5
5
  from .exprs import ExprBuilder
6
+ from .ops import OpKind
6
7
 
7
8
 
8
9
  ##
9
10
 
10
11
 
11
- class UnaryOp(Node, lang.Final):
12
+ class UnaryOp(dc.Frozen, lang.Final, eq=False):
12
13
  name: str
14
+ kind: OpKind
13
15
 
14
16
 
15
17
  class UnaryOps(lang.Namespace):
16
- NOT = UnaryOp('not')
18
+ NOT = UnaryOp('not', OpKind.CMP)
19
+ IS_NULL = UnaryOp('is_null', OpKind.CMP)
20
+ IS_NOT_NULL = UnaryOp('is_not_null', OpKind.CMP)
21
+
22
+ POS = UnaryOp('pos', OpKind.ARITH)
23
+ NEG = UnaryOp('neg', OpKind.ARITH)
17
24
 
18
25
 
19
26
  class Unary(Expr, lang.Final):
@@ -25,5 +32,21 @@ class UnaryBuilder(ExprBuilder):
25
32
  def unary(self, op: UnaryOp, v: CanExpr) -> Unary:
26
33
  return Unary(op, self.expr(v))
27
34
 
35
+ #
36
+
28
37
  def not_(self, v: CanExpr) -> Unary:
29
38
  return self.unary(UnaryOps.NOT, v)
39
+
40
+ def is_null(self, v: CanExpr) -> Unary:
41
+ return self.unary(UnaryOps.IS_NULL, v)
42
+
43
+ def is_not_null(self, v: CanExpr) -> Unary:
44
+ return self.unary(UnaryOps.IS_NOT_NULL, v)
45
+
46
+ #
47
+
48
+ def pos(self, v: CanExpr) -> Unary:
49
+ return self.unary(UnaryOps.POS, v)
50
+
51
+ def neg(self, v: CanExpr) -> Unary:
52
+ return self.unary(UnaryOps.NEG, v)
@@ -10,10 +10,7 @@ def _install_poly(cls: type) -> None:
10
10
  msh.STANDARD_UNMARSHALER_FACTORIES[0:0] = [msh.PolymorphismUnmarshalerFactory(p)]
11
11
 
12
12
 
13
- @lang.cached_function
13
+ @lang.static_init
14
14
  def _install_standard_marshalling() -> None:
15
15
  _install_poly(Dtype)
16
16
  _install_poly(Element)
17
-
18
-
19
- _install_standard_marshalling()
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: omlish
3
- Version: 0.0.0.dev72
3
+ Version: 0.0.0.dev74
4
4
  Summary: omlish
5
5
  Author: wrmsr
6
6
  License: BSD-3-Clause
@@ -1,10 +1,10 @@
1
1
  omlish/.manifests.json,sha256=TXvFdkAU0Zr2FKdo7fyvt9nr3UjCtrnAZ0diZXSAteE,1430
2
- omlish/__about__.py,sha256=Xtux9lxQlFpaTqHV9Xq7BumGnu8QP3C6l4R7wQX9F_w,3420
2
+ omlish/__about__.py,sha256=6L5yBogPnXXOqNId_oVRZJ4kjVB68VskkdZvIJ7NlE0,3420
3
3
  omlish/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
4
4
  omlish/argparse.py,sha256=Dc73G8lyoQBLvXhMYUbzQUh4SJu_OTvKUXjSUxq_ang,7499
5
5
  omlish/c3.py,sha256=4vogWgwPb8TbNS2KkZxpoWbwjj7MuHG2lQG-hdtkvjI,8062
6
6
  omlish/cached.py,sha256=UAizxlH4eMWHPzQtmItmyE6FEpFEUFzIkxaO2BHWZ5s,196
7
- omlish/check.py,sha256=fgWiBoHvqZlE8tjaxK7OMW4J8z3_rEGRENTka3EohbI,10378
7
+ omlish/check.py,sha256=rZFEn6IHiMq4KGCYxlUGcUCJP12pPPUs_-AG0aouPM8,10540
8
8
  omlish/datetimes.py,sha256=HajeM1kBvwlTa-uR1TTZHmZ3zTPnnUr1uGGQhiO1XQ0,2152
9
9
  omlish/defs.py,sha256=T3bq_7h_tO3nDB5RAFBn7DkdeQgqheXzkFColbOHZko,4890
10
10
  omlish/dynamic.py,sha256=35C_cCX_Vq2HrHzGk5T-zbrMvmUdiIiwDzDNixczoDo,6541
@@ -132,13 +132,13 @@ omlish/dataclasses/impl/as_.py,sha256=CD-t7hkC1EP2F_jvZKIA_cVoDuwZ-Ln_xC4fJumPYX
132
132
  omlish/dataclasses/impl/copy.py,sha256=Tn8_n6Vohs-w4otbGdubBEvhd3TsSTaM3EfNGdS2LYo,591
133
133
  omlish/dataclasses/impl/descriptors.py,sha256=rEYE1Len99agTQCC25hSPMnM19BgPr0ZChABGi58Fdk,2476
134
134
  omlish/dataclasses/impl/exceptions.py,sha256=DeiM6rcjgncudn-XVuph9TDbVDEwBtyYb1bcbO3FFcA,193
135
- omlish/dataclasses/impl/fields.py,sha256=iXA8j3q8GFQoj-6MgAKVLGG9GSSNOLGgyVtogCQ5q1E,6459
135
+ omlish/dataclasses/impl/fields.py,sha256=mr8tnSDceHGZ6VBbeegt-iCzQJbtCXoWOUwltjRULy4,6521
136
136
  omlish/dataclasses/impl/frozen.py,sha256=x87DSM8FIMZ3c_BIUE8NooCkExFjPsabeqIueEP5qKs,2988
137
137
  omlish/dataclasses/impl/hashing.py,sha256=FKnHuXCg9ylrzK2TLGqO5yfRN4HX3F415CSLlVYXtYE,3190
138
138
  omlish/dataclasses/impl/init.py,sha256=IgxO9nwHaHF8jGrUAk-Y5xke9uV2OwzfEe-88McE1Wg,6161
139
139
  omlish/dataclasses/impl/internals.py,sha256=UvZYjrLT1S8ntyxJ_vRPIkPOF00K8HatGAygErgoXTU,2990
140
140
  omlish/dataclasses/impl/main.py,sha256=Ti0PKbFKraKvfmoPuR-G7nLVNzRC8mvEuXhCuC-M2kc,2574
141
- omlish/dataclasses/impl/metaclass.py,sha256=E0lIKGjUcQXOJEZYb6qXa2h4iYNj292OFtau3sq8cSQ,3153
141
+ omlish/dataclasses/impl/metaclass.py,sha256=Fb0ExFiyYdOpvck4ayXMr_vEVDvHLhe28Ns3F4aduM8,3222
142
142
  omlish/dataclasses/impl/metadata.py,sha256=4veWwTr-aA0KP-Y1cPEeOcXHup9EKJTYNJ0ozIxtzD4,1401
143
143
  omlish/dataclasses/impl/order.py,sha256=zWvWDkSTym8cc7vO1cLHqcBhhjOlucHOCUVJcdh4jt0,1369
144
144
  omlish/dataclasses/impl/params.py,sha256=zsobAD6QvMAZeMsohP8nKFyv_w6Q4buocARQG1y1etA,2768
@@ -202,13 +202,13 @@ omlish/graphs/dot/rendering.py,sha256=2UgXvMRN4Z9cfIqLlC7Iu_8bWbwUDEL4opHHkFfSqT
202
202
  omlish/graphs/dot/utils.py,sha256=_FMwn77WfiiAfLsRTOKWm4IYbNv5kQN22YJ5psw6CWg,801
203
203
  omlish/http/__init__.py,sha256=-ENDALr8ehHvivRD6cxIbEC94t0RHhrakf6CQRDTc8o,625
204
204
  omlish/http/asgi.py,sha256=wXhBZ21bEl32Kv9yBrRwUR_7pHEgVtHP8ZZwbasQ6-4,3307
205
- omlish/http/clients.py,sha256=KPWDw251eBmiYSXw7KafMP9UQHEiLiLdh9ZLPV7jydc,6002
205
+ omlish/http/clients.py,sha256=eOY4bmbGdXuOOabt9NLAcTO7G49u85-HoAFW28mCXS4,6004
206
206
  omlish/http/collections.py,sha256=s8w5s4Gewgxxhe2Ai0R45PgJYYifrLgTbU3VXVflHj4,260
207
207
  omlish/http/consts.py,sha256=FTolezLknKU6WJjk_x2T3a5LEMlnZSqv7gzTq55lxcU,2147
208
208
  omlish/http/cookies.py,sha256=uuOYlHR6e2SC3GM41V0aozK10nef9tYg83Scqpn5-HM,6351
209
209
  omlish/http/dates.py,sha256=Otgp8wRxPgNGyzx8LFowu1vC4EKJYARCiAwLFncpfHM,2875
210
210
  omlish/http/encodings.py,sha256=w2WoKajpaZnQH8j-IBvk5ZFL2O2pAU_iBvZnkocaTlw,164
211
- omlish/http/headers.py,sha256=MO5uDwbViY6Z371Hl5OwTvh2DGl76_PnWHlurNwIsOs,4895
211
+ omlish/http/headers.py,sha256=S66wiXezBHybrnjAM15E9x4GJvPRvFQHeKaXdJ799fw,5028
212
212
  omlish/http/json.py,sha256=9XwAsl4966Mxrv-1ytyCqhcE6lbBJw-0_tFZzGszgHE,7440
213
213
  omlish/http/sessions.py,sha256=VZ_WS5uiQG5y7i3u8oKuQMqf8dPKUOjFm_qk_0OvI8c,4793
214
214
  omlish/http/wsgi.py,sha256=czZsVUX-l2YTlMrUjKN49wRoP4rVpS0qpeBn4O5BoMY,948
@@ -299,7 +299,7 @@ omlish/marshal/base.py,sha256=_ZKSSuv6p9r86sja_jaxI09WYpjCBFi0KTsOiZCYyk0,6587
299
299
  omlish/marshal/base64.py,sha256=F-3ogJdcFCtWINRgJgWT0rErqgx6f4qahhcg8OrkqhE,1089
300
300
  omlish/marshal/dataclasses.py,sha256=G6Uh8t4vvNBAx2BhzH8ksj8Hq_bo6npFphQhyF298VU,6892
301
301
  omlish/marshal/datetimes.py,sha256=0ffg8cEvx9SMKIXZGD9b7MqpLfmgw0uKKdn6YTfoqok,3714
302
- omlish/marshal/enums.py,sha256=-0fKutBbyz8ygEaA0_P_8IOJrI9jMGigmnPbutV9Bg4,1464
302
+ omlish/marshal/enums.py,sha256=CMAbx6RI2EcQoo7SoD-5q2l-3DFKreWMiOxs6mFpl_4,1472
303
303
  omlish/marshal/exceptions.py,sha256=jwQWn4LcPnadT2KRI_1JJCOSkwWh0yHnYK9BmSkNN4U,302
304
304
  omlish/marshal/factories.py,sha256=UV2Svjok-lTWsRqKGh-CeuAhvlohw9uJe7ZLyoKMvTM,2968
305
305
  omlish/marshal/forbidden.py,sha256=BNshzm4lN5O8sUZ1YvxrSYq3WPklq9NMQCRZ7RC3DLM,865
@@ -388,24 +388,27 @@ omlish/sql/alchemy/duckdb.py,sha256=kr7pIhiBLNAuZrcigHDtFg9zHkVcrRW3LfryO9VJ4mk,
388
388
  omlish/sql/alchemy/exprs.py,sha256=gO4Fj4xEY-PuDgV-N8hBMy55glZz7O-4H7v1LWabfZY,323
389
389
  omlish/sql/alchemy/secrets.py,sha256=EMfy4EfTbEvrlv_41oOhn8qsoF-eTkY7HciPenIE6rI,178
390
390
  omlish/sql/alchemy/sqlean.py,sha256=RbkuOuFIfM4fowwKk8-sQ6Dxk-tTUwxS94nY5Kxt52s,403
391
- omlish/sql/queries/__init__.py,sha256=TEl0iNWVxPURJIRJSaKpy5EJkyVXw2Zej8jlKmeggAo,963
391
+ omlish/sql/queries/__init__.py,sha256=BdhONMw2JebX-jEgYb1WrM5L3vkJOzQRm0nt04IYKgQ,1229
392
392
  omlish/sql/queries/base.py,sha256=_8O3MbH_OEjBnhp2oIJUZ3ClaQ8l4Sj9BdPdsP0Ie-g,224
393
- omlish/sql/queries/binary.py,sha256=7K8pX2ki4PsXo3RP-81LD-sl8A4nGAN7hbgJExwoQPw,1003
394
- omlish/sql/queries/exprs.py,sha256=tJ5gK21NKYbxKnJmrAUDkzaPqzg7hg6-z2GS4-MG05U,1092
393
+ omlish/sql/queries/binary.py,sha256=dcEzeEn104AMPuQ7QrJU2O-YCN3SUdxB5S4jaWKOUqY,2253
394
+ omlish/sql/queries/building.py,sha256=pddD8-WDDGwX97OW7MLahaZTLsO_0NLxxIIAXXq1N40,537
395
+ omlish/sql/queries/exprs.py,sha256=kAI5PmvfJ-TqEzzc9H4_womFShT1eA0jWSZH2j2Wg-c,1802
395
396
  omlish/sql/queries/idents.py,sha256=erW6fE9UapuvW1ZeLfGFz7yuW6zzktWIWmOuAHeF8_g,496
396
- omlish/sql/queries/multi.py,sha256=7qFq3-n3g6MmYHM_lyhru9DcabiZRAQUVaOyR3gXvXU,865
397
+ omlish/sql/queries/marshal.py,sha256=MKZ3CVcr7mpnmg25twVgErWk1SpRBEkE1mu2fE8QwUA,2508
398
+ omlish/sql/queries/multi.py,sha256=KKHf8uS3Akk-YACPUwaqQO20NlDdrB3gbauOAejZzgU,832
397
399
  omlish/sql/queries/names.py,sha256=YiIyS6ehYMYrdLlUxMawV_Xf2zdi7RwVO9Qsxr_W4_4,772
398
- omlish/sql/queries/relations.py,sha256=zAOCJWopA3MxAE_J4wxRtuvGEOcQgOnAQaF_wT4z930,804
399
- omlish/sql/queries/selects.py,sha256=yuFg_OZ0Nzn5Jgy3-gkkDV3JhtVrzR3Ov3asAzCgkQU,1328
400
- omlish/sql/queries/std.py,sha256=pddD8-WDDGwX97OW7MLahaZTLsO_0NLxxIIAXXq1N40,537
400
+ omlish/sql/queries/ops.py,sha256=B7IDfjr2DW5LJhWoNaY1WW90BJhe5ZtmxIELhWXbW-0,129
401
+ omlish/sql/queries/relations.py,sha256=-d3n-dN17c3TPMZmzSplhWawllUzdq12XPDAuzoeMEQ,838
402
+ omlish/sql/queries/rendering.py,sha256=AC2PSGFCLXYk38mnH7EAMV7hQOKp5x6O-ZYV8FXgnAM,4360
403
+ omlish/sql/queries/selects.py,sha256=EcHlyKl5kGSY1d3GVxnImhGCTB6WvwQnlSA9eZanBqU,1364
401
404
  omlish/sql/queries/stmts.py,sha256=pBqwD7dRlqMu6uh6vR3xaWOEgbZCcFWbOQ9ryYd17T4,441
402
- omlish/sql/queries/unary.py,sha256=B4PA__3GUB1L7bqG4e3fmUcS3Wypx8Ryy6P0rLiPBHQ,514
405
+ omlish/sql/queries/unary.py,sha256=MEYBDZn_H0bexmUrJeONOv5-gIpYowUaXOsEHeQM4ks,1144
403
406
  omlish/sql/tabledefs/__init__.py,sha256=TvtQsp-jJu6_ZahyCOFAaElSSBcRftNyJpdiDPGYCDk,190
404
407
  omlish/sql/tabledefs/alchemy.py,sha256=MiNfVSgX_Ka6PmVTgAcCmS3ULK5mfVRXD_VC2-9TidU,485
405
408
  omlish/sql/tabledefs/dtypes.py,sha256=egZDi-A17MC-4R_ZKR_3uQf1a6mGm91Gmh0b72O85bQ,362
406
409
  omlish/sql/tabledefs/elements.py,sha256=lP_Ch19hKmiGYPQVeC8HpFaKdTYnXi2FfpfwKMxZOck,1674
407
410
  omlish/sql/tabledefs/lower.py,sha256=YQf8gl1kxD5Fm-vOxV6G0Feh_D9PP1pYwz_vz6XjTPQ,1405
408
- omlish/sql/tabledefs/marshal.py,sha256=CvjNUHD5ABz72m7MsIqhpHyz2AZHYS6ijWwJXg3_MSs,542
411
+ omlish/sql/tabledefs/marshal.py,sha256=j-Rz1HsiXmABv39-2VoJdzSSB3kbxqaVevbdkZWMyG8,504
409
412
  omlish/sql/tabledefs/tabledefs.py,sha256=lIhvlt0pk6G7RZAtDFsFXm5j0l9BvRfnP7vNGeydHtE,816
410
413
  omlish/testing/__init__.py,sha256=M_BQrcCHkoL-ZvE-UpQ8XxXNYRRawhjUz4rCJnAqM2A,152
411
414
  omlish/testing/testing.py,sha256=TT2wwSzPZ_KhIvKxpM1qc1yHKD-LHDNgGrcr_h8vs7c,2895
@@ -433,9 +436,9 @@ omlish/text/delimit.py,sha256=ubPXcXQmtbOVrUsNh5gH1mDq5H-n1y2R4cPL5_DQf68,4928
433
436
  omlish/text/glyphsplit.py,sha256=Ug-dPRO7x-OrNNr8g1y6DotSZ2KH0S-VcOmUobwa4B0,3296
434
437
  omlish/text/indent.py,sha256=6Jj6TFY9unaPa4xPzrnZemJ-fHsV53IamP93XGjSUHs,1274
435
438
  omlish/text/parts.py,sha256=7vPF1aTZdvLVYJ4EwBZVzRSy8XB3YqPd7JwEnNGGAOo,6495
436
- omlish-0.0.0.dev72.dist-info/LICENSE,sha256=B_hVtavaA8zCYDW99DYdcpDLKz1n3BBRjZrcbv8uG8c,1451
437
- omlish-0.0.0.dev72.dist-info/METADATA,sha256=fC-RJQd1zeLsNghRMhMtxbjiAbWfSE6bbIBTkW85Rxg,4167
438
- omlish-0.0.0.dev72.dist-info/WHEEL,sha256=OVMc5UfuAQiSplgO0_WdW7vXVGAt9Hdd6qtN4HotdyA,91
439
- omlish-0.0.0.dev72.dist-info/entry_points.txt,sha256=Lt84WvRZJskWCAS7xnQGZIeVWksprtUHj0llrvVmod8,35
440
- omlish-0.0.0.dev72.dist-info/top_level.txt,sha256=pePsKdLu7DvtUiecdYXJ78iO80uDNmBlqe-8hOzOmfs,7
441
- omlish-0.0.0.dev72.dist-info/RECORD,,
439
+ omlish-0.0.0.dev74.dist-info/LICENSE,sha256=B_hVtavaA8zCYDW99DYdcpDLKz1n3BBRjZrcbv8uG8c,1451
440
+ omlish-0.0.0.dev74.dist-info/METADATA,sha256=fnVoKX28JDI8_XtMIkZGO22Po_cPDkZIi8nGwf6A_84,4167
441
+ omlish-0.0.0.dev74.dist-info/WHEEL,sha256=OVMc5UfuAQiSplgO0_WdW7vXVGAt9Hdd6qtN4HotdyA,91
442
+ omlish-0.0.0.dev74.dist-info/entry_points.txt,sha256=Lt84WvRZJskWCAS7xnQGZIeVWksprtUHj0llrvVmod8,35
443
+ omlish-0.0.0.dev74.dist-info/top_level.txt,sha256=pePsKdLu7DvtUiecdYXJ78iO80uDNmBlqe-8hOzOmfs,7
444
+ omlish-0.0.0.dev74.dist-info/RECORD,,
File without changes