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.
Files changed (62) hide show
  1. omlish/__about__.py +2 -2
  2. omlish/bootstrap/marshal.py +4 -3
  3. omlish/dataclasses/__init__.py +7 -0
  4. omlish/dataclasses/impl/metaclass.py +1 -0
  5. omlish/dataclasses/impl/simple.py +1 -1
  6. omlish/formats/toml/parser.py +593 -594
  7. omlish/lang/classes/abstract.py +3 -0
  8. omlish/lang/comparison.py +3 -0
  9. omlish/lang/datetimes.py +3 -0
  10. omlish/lang/generators.py +2 -2
  11. omlish/lang/maybes.py +3 -0
  12. omlish/lang/resolving.py +3 -0
  13. omlish/lang/sys.py +3 -0
  14. omlish/marshal/__init__.py +16 -0
  15. omlish/marshal/base.py +230 -14
  16. omlish/marshal/composite/iterables.py +3 -0
  17. omlish/marshal/composite/literals.py +7 -4
  18. omlish/marshal/composite/mappings.py +3 -0
  19. omlish/marshal/composite/maybes.py +3 -0
  20. omlish/marshal/composite/newtypes.py +7 -4
  21. omlish/marshal/composite/optionals.py +7 -4
  22. omlish/marshal/composite/wrapped.py +3 -0
  23. omlish/marshal/global_.py +16 -0
  24. omlish/marshal/objects/dataclasses.py +5 -5
  25. omlish/marshal/objects/marshal.py +2 -2
  26. omlish/marshal/objects/namedtuples.py +4 -4
  27. omlish/marshal/objects/unmarshal.py +2 -2
  28. omlish/marshal/polymorphism/marshal.py +2 -2
  29. omlish/marshal/polymorphism/metadata.py +9 -3
  30. omlish/marshal/polymorphism/unions.py +6 -6
  31. omlish/marshal/polymorphism/unmarshal.py +2 -2
  32. omlish/marshal/singular/base64.py +3 -0
  33. omlish/marshal/singular/enums.py +7 -4
  34. omlish/marshal/singular/numbers.py +3 -0
  35. omlish/marshal/singular/uuids.py +3 -0
  36. omlish/marshal/standard.py +25 -3
  37. omlish/marshal/trivial/any.py +3 -0
  38. omlish/marshal/trivial/forbidden.py +13 -2
  39. omlish/marshal/trivial/nop.py +3 -0
  40. omlish/secrets/marshal.py +4 -6
  41. omlish/specs/jsonrpc/marshal.py +4 -5
  42. omlish/specs/openapi/marshal.py +4 -2
  43. omlish/sql/alchemy/__init__.py +31 -0
  44. omlish/sql/alchemy/apiadapter.py +121 -0
  45. omlish/sql/api/__init__.py +39 -0
  46. omlish/sql/api/base.py +1 -0
  47. omlish/sql/parsing/parsing.py +1 -1
  48. omlish/sql/queries/__init__.py +4 -0
  49. omlish/sql/queries/base.py +113 -2
  50. omlish/sql/queries/exprs.py +15 -2
  51. omlish/sql/queries/inserts.py +2 -1
  52. omlish/sql/queries/marshal.py +23 -9
  53. omlish/sql/queries/params.py +3 -2
  54. omlish/sql/queries/rendering.py +16 -4
  55. omlish/sql/queries/selects.py +17 -2
  56. omlish/sql/tabledefs/marshal.py +4 -2
  57. {omlish-0.0.0.dev268.dist-info → omlish-0.0.0.dev270.dist-info}/METADATA +1 -1
  58. {omlish-0.0.0.dev268.dist-info → omlish-0.0.0.dev270.dist-info}/RECORD +62 -61
  59. {omlish-0.0.0.dev268.dist-info → omlish-0.0.0.dev270.dist-info}/WHEEL +0 -0
  60. {omlish-0.0.0.dev268.dist-info → omlish-0.0.0.dev270.dist-info}/entry_points.txt +0 -0
  61. {omlish-0.0.0.dev268.dist-info → omlish-0.0.0.dev270.dist-info}/licenses/LICENSE +0 -0
  62. {omlish-0.0.0.dev268.dist-info → omlish-0.0.0.dev270.dist-info}/top_level.txt +0 -0
@@ -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 Node(dc.Frozen, lang.Abstract, cache_hash=True):
17
- pass
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):
@@ -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:
@@ -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(dc.Frozen, lang.Final):
21
+ class Values(Node, lang.Final):
21
22
  vs: ta.Sequence[Expr] = dc.xfield(coerce=tuple)
22
23
 
23
24
 
@@ -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.STANDARD_MARSHALER_FACTORIES[0:0] = [msh.TypeMapMarshalerFactory({ty: OpMarshalerUnmarshaler(ty, ns)})]
68
- msh.STANDARD_UNMARSHALER_FACTORIES[0:0] = [msh.TypeMapUnmarshalerFactory({ty: OpMarshalerUnmarshaler(ty, ns)})]
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.STANDARD_MARSHALER_FACTORIES[0:0] = [msh.TypeMapMarshalerFactory({t: LowerEnumMarshaler(t) for t in ets})]
75
- msh.STANDARD_UNMARSHALER_FACTORIES[0:0] = [msh.TypeMapUnmarshalerFactory({t: LowerEnumMarshaler(t) for t in ets})]
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(cls, naming=msh.Naming.SNAKE)
84
- msh.STANDARD_MARSHALER_FACTORIES[0:0] = [msh.PolymorphismMarshalerFactory(p)]
85
- msh.STANDARD_UNMARSHALER_FACTORIES[0:0] = [msh.PolymorphismUnmarshalerFactory(p)]
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.STANDARD_MARSHALER_FACTORIES[0:0] = [msh.PolymorphismUnionMarshalerFactory(insert_data_impls)]
92
- msh.STANDARD_UNMARSHALER_FACTORIES[0:0] = [msh.PolymorphismUnionUnmarshalerFactory(insert_data_impls)]
103
+ msh.install_standard_factories(
104
+ msh.PolymorphismUnionMarshalerFactory(insert_data_impls),
105
+ msh.PolymorphismUnionUnmarshalerFactory(insert_data_impls),
106
+ )
@@ -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
- class Param(Expr, lang.Final):
11
+ @dc.dataclass(frozen=True)
12
+ class Param(lang.Final):
12
13
  n: str | None = None
13
14
 
14
15
  def __repr__(self) -> str:
@@ -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 render_param(self, o: Param) -> tp.Part:
153
- return self._params_preparer.add(o.n if o.n is not None else id(o))
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 render_select_item(self, o: SelectItem) -> tp.Part:
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 []),
@@ -16,11 +16,22 @@ from .stmts import Stmt
16
16
  ##
17
17
 
18
18
 
19
- class SelectItem(Node, lang.Final):
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 SelectItem(self.expr(o))
53
+ return ExprSelectItem(self.expr(o))
39
54
 
40
55
  def select(
41
56
  self,
@@ -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.STANDARD_MARSHALER_FACTORIES[0:0] = [msh.PolymorphismMarshalerFactory(p)]
10
- msh.STANDARD_UNMARSHALER_FACTORIES[0:0] = [msh.PolymorphismUnmarshalerFactory(p)]
9
+ msh.install_standard_factories(
10
+ msh.PolymorphismMarshalerFactory(p),
11
+ msh.PolymorphismUnmarshalerFactory(p),
12
+ )
11
13
 
12
14
 
13
15
  @lang.static_init
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: omlish
3
- Version: 0.0.0.dev268
3
+ Version: 0.0.0.dev270
4
4
  Summary: omlish
5
5
  Author: wrmsr
6
6
  License: BSD-3-Clause