omlish 0.0.0.dev268__py3-none-any.whl → 0.0.0.dev270__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/bootstrap/marshal.py +4 -3
- omlish/dataclasses/__init__.py +7 -0
- omlish/dataclasses/impl/metaclass.py +1 -0
- omlish/dataclasses/impl/simple.py +1 -1
- omlish/formats/toml/parser.py +593 -594
- omlish/lang/classes/abstract.py +3 -0
- omlish/lang/comparison.py +3 -0
- omlish/lang/datetimes.py +3 -0
- omlish/lang/generators.py +2 -2
- omlish/lang/maybes.py +3 -0
- omlish/lang/resolving.py +3 -0
- omlish/lang/sys.py +3 -0
- omlish/marshal/__init__.py +16 -0
- omlish/marshal/base.py +230 -14
- omlish/marshal/composite/iterables.py +3 -0
- omlish/marshal/composite/literals.py +7 -4
- omlish/marshal/composite/mappings.py +3 -0
- omlish/marshal/composite/maybes.py +3 -0
- omlish/marshal/composite/newtypes.py +7 -4
- omlish/marshal/composite/optionals.py +7 -4
- omlish/marshal/composite/wrapped.py +3 -0
- omlish/marshal/global_.py +16 -0
- omlish/marshal/objects/dataclasses.py +5 -5
- omlish/marshal/objects/marshal.py +2 -2
- omlish/marshal/objects/namedtuples.py +4 -4
- omlish/marshal/objects/unmarshal.py +2 -2
- omlish/marshal/polymorphism/marshal.py +2 -2
- omlish/marshal/polymorphism/metadata.py +9 -3
- omlish/marshal/polymorphism/unions.py +6 -6
- omlish/marshal/polymorphism/unmarshal.py +2 -2
- omlish/marshal/singular/base64.py +3 -0
- omlish/marshal/singular/enums.py +7 -4
- omlish/marshal/singular/numbers.py +3 -0
- omlish/marshal/singular/uuids.py +3 -0
- omlish/marshal/standard.py +25 -3
- omlish/marshal/trivial/any.py +3 -0
- omlish/marshal/trivial/forbidden.py +13 -2
- omlish/marshal/trivial/nop.py +3 -0
- omlish/secrets/marshal.py +4 -6
- omlish/specs/jsonrpc/marshal.py +4 -5
- omlish/specs/openapi/marshal.py +4 -2
- omlish/sql/alchemy/__init__.py +31 -0
- omlish/sql/alchemy/apiadapter.py +121 -0
- omlish/sql/api/__init__.py +39 -0
- omlish/sql/api/base.py +1 -0
- omlish/sql/parsing/parsing.py +1 -1
- omlish/sql/queries/__init__.py +4 -0
- omlish/sql/queries/base.py +113 -2
- omlish/sql/queries/exprs.py +15 -2
- omlish/sql/queries/inserts.py +2 -1
- omlish/sql/queries/marshal.py +23 -9
- omlish/sql/queries/params.py +3 -2
- omlish/sql/queries/rendering.py +16 -4
- omlish/sql/queries/selects.py +17 -2
- omlish/sql/tabledefs/marshal.py +4 -2
- {omlish-0.0.0.dev268.dist-info → omlish-0.0.0.dev270.dist-info}/METADATA +1 -1
- {omlish-0.0.0.dev268.dist-info → omlish-0.0.0.dev270.dist-info}/RECORD +62 -61
- {omlish-0.0.0.dev268.dist-info → omlish-0.0.0.dev270.dist-info}/WHEEL +0 -0
- {omlish-0.0.0.dev268.dist-info → omlish-0.0.0.dev270.dist-info}/entry_points.txt +0 -0
- {omlish-0.0.0.dev268.dist-info → omlish-0.0.0.dev270.dist-info}/licenses/LICENSE +0 -0
- {omlish-0.0.0.dev268.dist-info → omlish-0.0.0.dev270.dist-info}/top_level.txt +0 -0
omlish/sql/queries/base.py
CHANGED
@@ -1,3 +1,4 @@
|
|
1
|
+
import types
|
1
2
|
import typing as ta
|
2
3
|
|
3
4
|
from ... import dataclasses as dc
|
@@ -13,8 +14,118 @@ Value: ta.TypeAlias = ta.Any
|
|
13
14
|
##
|
14
15
|
|
15
16
|
|
16
|
-
class
|
17
|
-
|
17
|
+
class NodeComparisonTypeError(TypeError):
|
18
|
+
def __init__(self, cls: type['Node'], *args: ta.Any) -> None:
|
19
|
+
super().__init__(
|
20
|
+
'Query node types are not comparable or hashable - this would be a common source of confusion when '
|
21
|
+
'constructing query filters',
|
22
|
+
cls,
|
23
|
+
*args,
|
24
|
+
)
|
25
|
+
|
26
|
+
|
27
|
+
class Node(
|
28
|
+
dc.Frozen,
|
29
|
+
lang.Abstract,
|
30
|
+
eq=False,
|
31
|
+
confer=frozenset([
|
32
|
+
*dc.get_metaclass_params(dc.Frozen).confer,
|
33
|
+
'eq',
|
34
|
+
]),
|
35
|
+
):
|
36
|
+
@ta.final
|
37
|
+
def __hash__(self) -> ta.NoReturn:
|
38
|
+
raise NodeComparisonTypeError(type(self))
|
39
|
+
|
40
|
+
@ta.final
|
41
|
+
def __eq__(self, other) -> ta.NoReturn:
|
42
|
+
raise NodeComparisonTypeError(type(self))
|
43
|
+
|
44
|
+
@ta.final
|
45
|
+
def __ne__(self, other) -> ta.NoReturn:
|
46
|
+
raise NodeComparisonTypeError(type(self))
|
47
|
+
|
48
|
+
#
|
49
|
+
|
50
|
+
@dc.dataclass(frozen=True)
|
51
|
+
class _Fields:
|
52
|
+
cmp_fields: ta.Sequence[str]
|
53
|
+
hash_fields: ta.Sequence[str]
|
54
|
+
|
55
|
+
__node_fields__: ta.ClassVar[tuple[str, ...]]
|
56
|
+
|
57
|
+
@classmethod
|
58
|
+
def _fields(cls) -> _Fields:
|
59
|
+
try:
|
60
|
+
return cls.__dict__['__node_fields__']
|
61
|
+
except KeyError:
|
62
|
+
pass
|
63
|
+
|
64
|
+
dc_info = dc.reflect(cls)
|
65
|
+
fields = Node._Fields(
|
66
|
+
cmp_fields=tuple(f.name for f in dc_info.instance_fields if f.compare),
|
67
|
+
hash_fields=tuple(f.name for f in dc_info.instance_fields if (f.compare if f.hash is None else f.hash)),
|
68
|
+
)
|
69
|
+
|
70
|
+
setattr(cls, '__node_fields__', fields)
|
71
|
+
return fields
|
72
|
+
|
73
|
+
_hash: ta.ClassVar[int]
|
74
|
+
|
75
|
+
def hash(self) -> int:
|
76
|
+
try:
|
77
|
+
return self._hash
|
78
|
+
except AttributeError:
|
79
|
+
pass
|
80
|
+
|
81
|
+
h = hash(tuple(getattr(self, f) for f in self._fields().hash_fields))
|
82
|
+
object.__setattr__(self, '_hash', h)
|
83
|
+
return h
|
84
|
+
|
85
|
+
def eq(self, other: 'Node') -> bool | types.NotImplementedType:
|
86
|
+
if self is other:
|
87
|
+
return True
|
88
|
+
|
89
|
+
def rec(l, r):
|
90
|
+
if l.__class__ is not r.__class__:
|
91
|
+
return NotImplemented
|
92
|
+
|
93
|
+
if isinstance(l, lang.BUILTIN_SCALAR_ITERABLE_TYPES):
|
94
|
+
return l == r
|
95
|
+
|
96
|
+
elif isinstance(l, ta.Mapping):
|
97
|
+
ks = set(l)
|
98
|
+
if set(r) != ks:
|
99
|
+
return False
|
100
|
+
for k, lv in l.items():
|
101
|
+
rv = r[k]
|
102
|
+
if (ret := rec(lv, rv)) is not True:
|
103
|
+
return ret
|
104
|
+
return True
|
105
|
+
|
106
|
+
elif isinstance(l, ta.Sequence):
|
107
|
+
if len(l) != len(r):
|
108
|
+
return False
|
109
|
+
for le, re in zip(l, r):
|
110
|
+
if (ret := rec(le, re)) is not True:
|
111
|
+
return ret
|
112
|
+
return True
|
113
|
+
|
114
|
+
elif isinstance(l, Node):
|
115
|
+
return l.eq(r)
|
116
|
+
|
117
|
+
else:
|
118
|
+
return l == r
|
119
|
+
|
120
|
+
for f in self._fields().cmp_fields:
|
121
|
+
lf = getattr(self, f)
|
122
|
+
rf = getattr(other, f)
|
123
|
+
if (ret := rec(lf, rf)) is not True:
|
124
|
+
return ret
|
125
|
+
return True
|
126
|
+
|
127
|
+
|
128
|
+
##
|
18
129
|
|
19
130
|
|
20
131
|
class Builder(lang.Abstract):
|
omlish/sql/queries/exprs.py
CHANGED
@@ -2,6 +2,7 @@
|
|
2
2
|
TODO:
|
3
3
|
- case
|
4
4
|
- cast / ::
|
5
|
+
- explicit null? already wrapped in non-null Literal node
|
5
6
|
"""
|
6
7
|
import typing as ta
|
7
8
|
|
@@ -13,6 +14,9 @@ from .names import CanName
|
|
13
14
|
from .names import Name
|
14
15
|
from .names import NameBuilder
|
15
16
|
from .names import NameLike
|
17
|
+
from .params import CanParam
|
18
|
+
from .params import Param
|
19
|
+
from .params import ParamBuilder
|
16
20
|
|
17
21
|
|
18
22
|
##
|
@@ -36,14 +40,21 @@ class NameExpr(Expr, lang.Final):
|
|
36
40
|
n: Name
|
37
41
|
|
38
42
|
|
43
|
+
#
|
44
|
+
|
45
|
+
|
46
|
+
class ParamExpr(Expr, lang.Final):
|
47
|
+
p: Param
|
48
|
+
|
49
|
+
|
39
50
|
##
|
40
51
|
|
41
52
|
|
42
53
|
CanLiteral: ta.TypeAlias = Literal | Value
|
43
|
-
CanExpr: ta.TypeAlias = Expr | CanName | CanLiteral
|
54
|
+
CanExpr: ta.TypeAlias = Expr | CanParam | CanName | CanLiteral
|
44
55
|
|
45
56
|
|
46
|
-
class ExprBuilder(NameBuilder):
|
57
|
+
class ExprBuilder(ParamBuilder, NameBuilder):
|
47
58
|
def literal(self, o: CanLiteral) -> Literal:
|
48
59
|
if isinstance(o, Literal):
|
49
60
|
return o
|
@@ -61,6 +72,8 @@ class ExprBuilder(NameBuilder):
|
|
61
72
|
def expr(self, o: CanExpr) -> Expr:
|
62
73
|
if isinstance(o, Expr):
|
63
74
|
return o
|
75
|
+
elif isinstance(o, Param):
|
76
|
+
return ParamExpr(o)
|
64
77
|
elif isinstance(o, (NameLike, IdentLike)):
|
65
78
|
return NameExpr(self.name(o))
|
66
79
|
else:
|
omlish/sql/queries/inserts.py
CHANGED
@@ -2,6 +2,7 @@ import typing as ta
|
|
2
2
|
|
3
3
|
from ... import dataclasses as dc
|
4
4
|
from ... import lang
|
5
|
+
from .base import Node
|
5
6
|
from .exprs import CanExpr
|
6
7
|
from .exprs import Expr
|
7
8
|
from .exprs import ExprBuilder
|
@@ -17,7 +18,7 @@ from .stmts import Stmt
|
|
17
18
|
##
|
18
19
|
|
19
20
|
|
20
|
-
class Values(
|
21
|
+
class Values(Node, lang.Final):
|
21
22
|
vs: ta.Sequence[Expr] = dc.xfield(coerce=tuple)
|
22
23
|
|
23
24
|
|
omlish/sql/queries/marshal.py
CHANGED
@@ -20,6 +20,7 @@ from .multi import MultiKind
|
|
20
20
|
from .relations import JoinKind
|
21
21
|
from .relations import Relation
|
22
22
|
from .selects import Select
|
23
|
+
from .selects import SelectItem
|
23
24
|
from .stmts import Stmt
|
24
25
|
from .unary import UnaryOp
|
25
26
|
from .unary import UnaryOps
|
@@ -64,29 +65,42 @@ def _install_standard_marshalling() -> None:
|
|
64
65
|
(BinaryOp, BinaryOps),
|
65
66
|
(UnaryOp, UnaryOps),
|
66
67
|
]:
|
67
|
-
msh.
|
68
|
-
|
68
|
+
msh.install_standard_factories(
|
69
|
+
msh.TypeMapMarshalerFactory({ty: OpMarshalerUnmarshaler(ty, ns)}),
|
70
|
+
msh.TypeMapUnmarshalerFactory({ty: OpMarshalerUnmarshaler(ty, ns)}),
|
71
|
+
)
|
69
72
|
|
70
73
|
ets = [
|
71
74
|
JoinKind,
|
72
75
|
MultiKind,
|
73
76
|
]
|
74
|
-
msh.
|
75
|
-
|
77
|
+
msh.install_standard_factories(
|
78
|
+
msh.TypeMapMarshalerFactory({t: LowerEnumMarshaler(t) for t in ets}),
|
79
|
+
msh.TypeMapUnmarshalerFactory({t: LowerEnumMarshaler(t) for t in ets}),
|
80
|
+
)
|
76
81
|
|
77
82
|
for cls in [
|
78
83
|
Expr,
|
79
84
|
Node,
|
80
85
|
Relation,
|
86
|
+
SelectItem,
|
81
87
|
Stmt,
|
82
88
|
]:
|
83
|
-
p = msh.polymorphism_from_subclasses(
|
84
|
-
|
85
|
-
|
89
|
+
p = msh.polymorphism_from_subclasses(
|
90
|
+
cls,
|
91
|
+
naming=msh.Naming.SNAKE,
|
92
|
+
strip_suffix='auto',
|
93
|
+
)
|
94
|
+
msh.install_standard_factories(
|
95
|
+
msh.PolymorphismMarshalerFactory(p),
|
96
|
+
msh.PolymorphismUnmarshalerFactory(p),
|
97
|
+
)
|
86
98
|
|
87
99
|
insert_data_impls = msh.Impls([
|
88
100
|
msh.Impl(Values, 'values'),
|
89
101
|
msh.Impl(Select, 'select'),
|
90
102
|
])
|
91
|
-
msh.
|
92
|
-
|
103
|
+
msh.install_standard_factories(
|
104
|
+
msh.PolymorphismUnionMarshalerFactory(insert_data_impls),
|
105
|
+
msh.PolymorphismUnionUnmarshalerFactory(insert_data_impls),
|
106
|
+
)
|
omlish/sql/queries/params.py
CHANGED
@@ -1,14 +1,15 @@
|
|
1
1
|
import typing as ta
|
2
2
|
|
3
|
+
from ... import dataclasses as dc
|
3
4
|
from ... import lang
|
4
5
|
from .base import Builder
|
5
|
-
from .exprs import Expr
|
6
6
|
|
7
7
|
|
8
8
|
##
|
9
9
|
|
10
10
|
|
11
|
-
|
11
|
+
@dc.dataclass(frozen=True)
|
12
|
+
class Param(lang.Final):
|
12
13
|
n: str | None = None
|
13
14
|
|
14
15
|
def __repr__(self) -> str:
|
omlish/sql/queries/rendering.py
CHANGED
@@ -29,6 +29,7 @@ from .binary import BinaryOp
|
|
29
29
|
from .binary import BinaryOps
|
30
30
|
from .exprs import Literal
|
31
31
|
from .exprs import NameExpr
|
32
|
+
from .exprs import ParamExpr
|
32
33
|
from .idents import Ident
|
33
34
|
from .inserts import Insert
|
34
35
|
from .inserts import Values
|
@@ -39,8 +40,9 @@ from .params import Param
|
|
39
40
|
from .relations import Join
|
40
41
|
from .relations import JoinKind
|
41
42
|
from .relations import Table
|
43
|
+
from .selects import AllSelectItem
|
44
|
+
from .selects import ExprSelectItem
|
42
45
|
from .selects import Select
|
43
|
-
from .selects import SelectItem
|
44
46
|
from .unary import Unary
|
45
47
|
from .unary import UnaryOp
|
46
48
|
from .unary import UnaryOps
|
@@ -149,8 +151,8 @@ class StdRenderer(Renderer):
|
|
149
151
|
return self.render(o.n)
|
150
152
|
|
151
153
|
@Renderer.render.register
|
152
|
-
def
|
153
|
-
return self.
|
154
|
+
def render_param_expr(self, o: ParamExpr) -> tp.Part:
|
155
|
+
return self.render(o.p)
|
154
156
|
|
155
157
|
# idents
|
156
158
|
|
@@ -201,6 +203,12 @@ class StdRenderer(Renderer):
|
|
201
203
|
out.append(self.render(i))
|
202
204
|
return tp.Concat(out)
|
203
205
|
|
206
|
+
# params
|
207
|
+
|
208
|
+
@Renderer.render.register
|
209
|
+
def render_param(self, o: Param) -> tp.Part:
|
210
|
+
return self._params_preparer.add(o.n if o.n is not None else id(o))
|
211
|
+
|
204
212
|
# relations
|
205
213
|
|
206
214
|
@Renderer.render.register
|
@@ -235,7 +243,11 @@ class StdRenderer(Renderer):
|
|
235
243
|
# selects
|
236
244
|
|
237
245
|
@Renderer.render.register
|
238
|
-
def
|
246
|
+
def render_all_select_item(self, o: AllSelectItem) -> tp.Part:
|
247
|
+
return '*'
|
248
|
+
|
249
|
+
@Renderer.render.register
|
250
|
+
def render_expr_select_item(self, o: ExprSelectItem) -> tp.Part:
|
239
251
|
return [
|
240
252
|
self.render(o.v),
|
241
253
|
*(['as', self.render(o.a)] if o.a is not None else []),
|
omlish/sql/queries/selects.py
CHANGED
@@ -16,11 +16,22 @@ from .stmts import Stmt
|
|
16
16
|
##
|
17
17
|
|
18
18
|
|
19
|
-
class SelectItem(Node, lang.
|
19
|
+
class SelectItem(Node, lang.Abstract):
|
20
|
+
pass
|
21
|
+
|
22
|
+
|
23
|
+
class AllSelectItem(SelectItem, lang.Final):
|
24
|
+
pass
|
25
|
+
|
26
|
+
|
27
|
+
class ExprSelectItem(SelectItem, lang.Final):
|
20
28
|
v: Expr
|
21
29
|
a: Ident | None = dc.xfield(None, repr_fn=dc.opt_repr)
|
22
30
|
|
23
31
|
|
32
|
+
##
|
33
|
+
|
34
|
+
|
24
35
|
class Select(Stmt, lang.Final):
|
25
36
|
items: ta.Sequence[SelectItem] = dc.xfield(coerce=tuple)
|
26
37
|
from_: Relation | None = dc.xfield(None, repr_fn=dc.opt_repr)
|
@@ -31,11 +42,15 @@ CanSelectItem: ta.TypeAlias = SelectItem | CanExpr
|
|
31
42
|
|
32
43
|
|
33
44
|
class SelectBuilder(RelationBuilder, ExprBuilder):
|
45
|
+
@property
|
46
|
+
def star(self) -> AllSelectItem:
|
47
|
+
return AllSelectItem()
|
48
|
+
|
34
49
|
def select_item(self, o: CanSelectItem) -> SelectItem:
|
35
50
|
if isinstance(o, SelectItem):
|
36
51
|
return o
|
37
52
|
else:
|
38
|
-
return
|
53
|
+
return ExprSelectItem(self.expr(o))
|
39
54
|
|
40
55
|
def select(
|
41
56
|
self,
|
omlish/sql/tabledefs/marshal.py
CHANGED
@@ -6,8 +6,10 @@ from .elements import Element
|
|
6
6
|
|
7
7
|
def _install_poly(cls: type) -> None:
|
8
8
|
p = msh.polymorphism_from_subclasses(cls, naming=msh.Naming.SNAKE)
|
9
|
-
msh.
|
10
|
-
|
9
|
+
msh.install_standard_factories(
|
10
|
+
msh.PolymorphismMarshalerFactory(p),
|
11
|
+
msh.PolymorphismUnmarshalerFactory(p),
|
12
|
+
)
|
11
13
|
|
12
14
|
|
13
15
|
@lang.static_init
|