SQLAlchemy 2.1.0b2__cp313-cp313t-win_arm64.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.
- sqlalchemy/__init__.py +298 -0
- sqlalchemy/connectors/__init__.py +18 -0
- sqlalchemy/connectors/aioodbc.py +171 -0
- sqlalchemy/connectors/asyncio.py +476 -0
- sqlalchemy/connectors/pyodbc.py +250 -0
- sqlalchemy/dialects/__init__.py +62 -0
- sqlalchemy/dialects/_typing.py +30 -0
- sqlalchemy/dialects/mssql/__init__.py +89 -0
- sqlalchemy/dialects/mssql/aioodbc.py +63 -0
- sqlalchemy/dialects/mssql/base.py +4166 -0
- sqlalchemy/dialects/mssql/information_schema.py +285 -0
- sqlalchemy/dialects/mssql/json.py +140 -0
- sqlalchemy/dialects/mssql/mssqlpython.py +220 -0
- sqlalchemy/dialects/mssql/provision.py +196 -0
- sqlalchemy/dialects/mssql/pymssql.py +126 -0
- sqlalchemy/dialects/mssql/pyodbc.py +698 -0
- sqlalchemy/dialects/mysql/__init__.py +106 -0
- sqlalchemy/dialects/mysql/_mariadb_shim.py +312 -0
- sqlalchemy/dialects/mysql/aiomysql.py +226 -0
- sqlalchemy/dialects/mysql/asyncmy.py +214 -0
- sqlalchemy/dialects/mysql/base.py +3877 -0
- sqlalchemy/dialects/mysql/cymysql.py +106 -0
- sqlalchemy/dialects/mysql/dml.py +279 -0
- sqlalchemy/dialects/mysql/enumerated.py +277 -0
- sqlalchemy/dialects/mysql/expression.py +146 -0
- sqlalchemy/dialects/mysql/json.py +92 -0
- sqlalchemy/dialects/mysql/mariadb.py +67 -0
- sqlalchemy/dialects/mysql/mariadbconnector.py +330 -0
- sqlalchemy/dialects/mysql/mysqlconnector.py +296 -0
- sqlalchemy/dialects/mysql/mysqldb.py +312 -0
- sqlalchemy/dialects/mysql/provision.py +153 -0
- sqlalchemy/dialects/mysql/pymysql.py +157 -0
- sqlalchemy/dialects/mysql/pyodbc.py +156 -0
- sqlalchemy/dialects/mysql/reflection.py +724 -0
- sqlalchemy/dialects/mysql/reserved_words.py +570 -0
- sqlalchemy/dialects/mysql/types.py +845 -0
- sqlalchemy/dialects/oracle/__init__.py +85 -0
- sqlalchemy/dialects/oracle/base.py +3977 -0
- sqlalchemy/dialects/oracle/cx_oracle.py +1601 -0
- sqlalchemy/dialects/oracle/dictionary.py +507 -0
- sqlalchemy/dialects/oracle/json.py +158 -0
- sqlalchemy/dialects/oracle/oracledb.py +909 -0
- sqlalchemy/dialects/oracle/provision.py +288 -0
- sqlalchemy/dialects/oracle/types.py +367 -0
- sqlalchemy/dialects/oracle/vector.py +368 -0
- sqlalchemy/dialects/postgresql/__init__.py +171 -0
- sqlalchemy/dialects/postgresql/_psycopg_common.py +229 -0
- sqlalchemy/dialects/postgresql/array.py +534 -0
- sqlalchemy/dialects/postgresql/asyncpg.py +1323 -0
- sqlalchemy/dialects/postgresql/base.py +5789 -0
- sqlalchemy/dialects/postgresql/bitstring.py +327 -0
- sqlalchemy/dialects/postgresql/dml.py +360 -0
- sqlalchemy/dialects/postgresql/ext.py +593 -0
- sqlalchemy/dialects/postgresql/hstore.py +423 -0
- sqlalchemy/dialects/postgresql/json.py +408 -0
- sqlalchemy/dialects/postgresql/named_types.py +521 -0
- sqlalchemy/dialects/postgresql/operators.py +130 -0
- sqlalchemy/dialects/postgresql/pg8000.py +670 -0
- sqlalchemy/dialects/postgresql/pg_catalog.py +344 -0
- sqlalchemy/dialects/postgresql/provision.py +184 -0
- sqlalchemy/dialects/postgresql/psycopg.py +799 -0
- sqlalchemy/dialects/postgresql/psycopg2.py +860 -0
- sqlalchemy/dialects/postgresql/psycopg2cffi.py +61 -0
- sqlalchemy/dialects/postgresql/ranges.py +1002 -0
- sqlalchemy/dialects/postgresql/types.py +388 -0
- sqlalchemy/dialects/sqlite/__init__.py +57 -0
- sqlalchemy/dialects/sqlite/aiosqlite.py +321 -0
- sqlalchemy/dialects/sqlite/base.py +3063 -0
- sqlalchemy/dialects/sqlite/dml.py +279 -0
- sqlalchemy/dialects/sqlite/json.py +100 -0
- sqlalchemy/dialects/sqlite/provision.py +229 -0
- sqlalchemy/dialects/sqlite/pysqlcipher.py +161 -0
- sqlalchemy/dialects/sqlite/pysqlite.py +754 -0
- sqlalchemy/dialects/type_migration_guidelines.txt +145 -0
- sqlalchemy/engine/__init__.py +62 -0
- sqlalchemy/engine/_processors_cy.cp313t-win_arm64.pyd +0 -0
- sqlalchemy/engine/_processors_cy.py +92 -0
- sqlalchemy/engine/_result_cy.cp313t-win_arm64.pyd +0 -0
- sqlalchemy/engine/_result_cy.py +633 -0
- sqlalchemy/engine/_row_cy.cp313t-win_arm64.pyd +0 -0
- sqlalchemy/engine/_row_cy.py +232 -0
- sqlalchemy/engine/_util_cy.cp313t-win_arm64.pyd +0 -0
- sqlalchemy/engine/_util_cy.py +136 -0
- sqlalchemy/engine/base.py +3354 -0
- sqlalchemy/engine/characteristics.py +155 -0
- sqlalchemy/engine/create.py +877 -0
- sqlalchemy/engine/cursor.py +2421 -0
- sqlalchemy/engine/default.py +2402 -0
- sqlalchemy/engine/events.py +965 -0
- sqlalchemy/engine/interfaces.py +3495 -0
- sqlalchemy/engine/mock.py +134 -0
- sqlalchemy/engine/processors.py +82 -0
- sqlalchemy/engine/reflection.py +2100 -0
- sqlalchemy/engine/result.py +1966 -0
- sqlalchemy/engine/row.py +397 -0
- sqlalchemy/engine/strategies.py +16 -0
- sqlalchemy/engine/url.py +922 -0
- sqlalchemy/engine/util.py +156 -0
- sqlalchemy/event/__init__.py +26 -0
- sqlalchemy/event/api.py +220 -0
- sqlalchemy/event/attr.py +674 -0
- sqlalchemy/event/base.py +472 -0
- sqlalchemy/event/legacy.py +258 -0
- sqlalchemy/event/registry.py +390 -0
- sqlalchemy/events.py +17 -0
- sqlalchemy/exc.py +922 -0
- sqlalchemy/ext/__init__.py +11 -0
- sqlalchemy/ext/associationproxy.py +2072 -0
- sqlalchemy/ext/asyncio/__init__.py +29 -0
- sqlalchemy/ext/asyncio/base.py +281 -0
- sqlalchemy/ext/asyncio/engine.py +1487 -0
- sqlalchemy/ext/asyncio/exc.py +21 -0
- sqlalchemy/ext/asyncio/result.py +994 -0
- sqlalchemy/ext/asyncio/scoping.py +1679 -0
- sqlalchemy/ext/asyncio/session.py +2007 -0
- sqlalchemy/ext/automap.py +1701 -0
- sqlalchemy/ext/baked.py +559 -0
- sqlalchemy/ext/compiler.py +600 -0
- sqlalchemy/ext/declarative/__init__.py +65 -0
- sqlalchemy/ext/declarative/extensions.py +560 -0
- sqlalchemy/ext/horizontal_shard.py +481 -0
- sqlalchemy/ext/hybrid.py +1877 -0
- sqlalchemy/ext/indexable.py +364 -0
- sqlalchemy/ext/instrumentation.py +450 -0
- sqlalchemy/ext/mutable.py +1081 -0
- sqlalchemy/ext/orderinglist.py +439 -0
- sqlalchemy/ext/serializer.py +185 -0
- sqlalchemy/future/__init__.py +16 -0
- sqlalchemy/future/engine.py +15 -0
- sqlalchemy/inspection.py +174 -0
- sqlalchemy/log.py +283 -0
- sqlalchemy/orm/__init__.py +176 -0
- sqlalchemy/orm/_orm_constructors.py +2694 -0
- sqlalchemy/orm/_typing.py +179 -0
- sqlalchemy/orm/attributes.py +2868 -0
- sqlalchemy/orm/base.py +976 -0
- sqlalchemy/orm/bulk_persistence.py +2152 -0
- sqlalchemy/orm/clsregistry.py +582 -0
- sqlalchemy/orm/collections.py +1568 -0
- sqlalchemy/orm/context.py +3471 -0
- sqlalchemy/orm/decl_api.py +2280 -0
- sqlalchemy/orm/decl_base.py +2309 -0
- sqlalchemy/orm/dependency.py +1306 -0
- sqlalchemy/orm/descriptor_props.py +1183 -0
- sqlalchemy/orm/dynamic.py +307 -0
- sqlalchemy/orm/evaluator.py +379 -0
- sqlalchemy/orm/events.py +3386 -0
- sqlalchemy/orm/exc.py +237 -0
- sqlalchemy/orm/identity.py +302 -0
- sqlalchemy/orm/instrumentation.py +746 -0
- sqlalchemy/orm/interfaces.py +1589 -0
- sqlalchemy/orm/loading.py +1684 -0
- sqlalchemy/orm/mapped_collection.py +557 -0
- sqlalchemy/orm/mapper.py +4411 -0
- sqlalchemy/orm/path_registry.py +829 -0
- sqlalchemy/orm/persistence.py +1789 -0
- sqlalchemy/orm/properties.py +973 -0
- sqlalchemy/orm/query.py +3528 -0
- sqlalchemy/orm/relationships.py +3570 -0
- sqlalchemy/orm/scoping.py +2232 -0
- sqlalchemy/orm/session.py +5403 -0
- sqlalchemy/orm/state.py +1175 -0
- sqlalchemy/orm/state_changes.py +196 -0
- sqlalchemy/orm/strategies.py +3492 -0
- sqlalchemy/orm/strategy_options.py +2562 -0
- sqlalchemy/orm/sync.py +164 -0
- sqlalchemy/orm/unitofwork.py +798 -0
- sqlalchemy/orm/util.py +2438 -0
- sqlalchemy/orm/writeonly.py +694 -0
- sqlalchemy/pool/__init__.py +41 -0
- sqlalchemy/pool/base.py +1522 -0
- sqlalchemy/pool/events.py +375 -0
- sqlalchemy/pool/impl.py +582 -0
- sqlalchemy/py.typed +0 -0
- sqlalchemy/schema.py +74 -0
- sqlalchemy/sql/__init__.py +156 -0
- sqlalchemy/sql/_annotated_cols.py +397 -0
- sqlalchemy/sql/_dml_constructors.py +132 -0
- sqlalchemy/sql/_elements_constructors.py +2164 -0
- sqlalchemy/sql/_orm_types.py +20 -0
- sqlalchemy/sql/_selectable_constructors.py +840 -0
- sqlalchemy/sql/_typing.py +487 -0
- sqlalchemy/sql/_util_cy.cp313t-win_arm64.pyd +0 -0
- sqlalchemy/sql/_util_cy.py +127 -0
- sqlalchemy/sql/annotation.py +590 -0
- sqlalchemy/sql/base.py +2699 -0
- sqlalchemy/sql/cache_key.py +1066 -0
- sqlalchemy/sql/coercions.py +1373 -0
- sqlalchemy/sql/compiler.py +8327 -0
- sqlalchemy/sql/crud.py +1815 -0
- sqlalchemy/sql/ddl.py +1928 -0
- sqlalchemy/sql/default_comparator.py +654 -0
- sqlalchemy/sql/dml.py +1977 -0
- sqlalchemy/sql/elements.py +6033 -0
- sqlalchemy/sql/events.py +458 -0
- sqlalchemy/sql/expression.py +172 -0
- sqlalchemy/sql/functions.py +2305 -0
- sqlalchemy/sql/lambdas.py +1443 -0
- sqlalchemy/sql/naming.py +209 -0
- sqlalchemy/sql/operators.py +2897 -0
- sqlalchemy/sql/roles.py +332 -0
- sqlalchemy/sql/schema.py +6703 -0
- sqlalchemy/sql/selectable.py +7553 -0
- sqlalchemy/sql/sqltypes.py +4093 -0
- sqlalchemy/sql/traversals.py +1042 -0
- sqlalchemy/sql/type_api.py +2446 -0
- sqlalchemy/sql/util.py +1495 -0
- sqlalchemy/sql/visitors.py +1157 -0
- sqlalchemy/testing/__init__.py +96 -0
- sqlalchemy/testing/assertions.py +1007 -0
- sqlalchemy/testing/assertsql.py +519 -0
- sqlalchemy/testing/asyncio.py +128 -0
- sqlalchemy/testing/config.py +440 -0
- sqlalchemy/testing/engines.py +483 -0
- sqlalchemy/testing/entities.py +117 -0
- sqlalchemy/testing/exclusions.py +476 -0
- sqlalchemy/testing/fixtures/__init__.py +30 -0
- sqlalchemy/testing/fixtures/base.py +384 -0
- sqlalchemy/testing/fixtures/mypy.py +247 -0
- sqlalchemy/testing/fixtures/orm.py +227 -0
- sqlalchemy/testing/fixtures/sql.py +538 -0
- sqlalchemy/testing/pickleable.py +155 -0
- sqlalchemy/testing/plugin/__init__.py +6 -0
- sqlalchemy/testing/plugin/bootstrap.py +51 -0
- sqlalchemy/testing/plugin/plugin_base.py +828 -0
- sqlalchemy/testing/plugin/pytestplugin.py +892 -0
- sqlalchemy/testing/profiling.py +329 -0
- sqlalchemy/testing/provision.py +613 -0
- sqlalchemy/testing/requirements.py +1978 -0
- sqlalchemy/testing/schema.py +198 -0
- sqlalchemy/testing/suite/__init__.py +19 -0
- sqlalchemy/testing/suite/test_cte.py +237 -0
- sqlalchemy/testing/suite/test_ddl.py +420 -0
- sqlalchemy/testing/suite/test_dialect.py +776 -0
- sqlalchemy/testing/suite/test_insert.py +630 -0
- sqlalchemy/testing/suite/test_reflection.py +3557 -0
- sqlalchemy/testing/suite/test_results.py +660 -0
- sqlalchemy/testing/suite/test_rowcount.py +258 -0
- sqlalchemy/testing/suite/test_select.py +2112 -0
- sqlalchemy/testing/suite/test_sequence.py +317 -0
- sqlalchemy/testing/suite/test_table_via_select.py +686 -0
- sqlalchemy/testing/suite/test_types.py +2271 -0
- sqlalchemy/testing/suite/test_unicode_ddl.py +189 -0
- sqlalchemy/testing/suite/test_update_delete.py +139 -0
- sqlalchemy/testing/util.py +535 -0
- sqlalchemy/testing/warnings.py +52 -0
- sqlalchemy/types.py +76 -0
- sqlalchemy/util/__init__.py +158 -0
- sqlalchemy/util/_collections.py +688 -0
- sqlalchemy/util/_collections_cy.cp313t-win_arm64.pyd +0 -0
- sqlalchemy/util/_collections_cy.pxd +8 -0
- sqlalchemy/util/_collections_cy.py +516 -0
- sqlalchemy/util/_has_cython.py +46 -0
- sqlalchemy/util/_immutabledict_cy.cp313t-win_arm64.pyd +0 -0
- sqlalchemy/util/_immutabledict_cy.py +240 -0
- sqlalchemy/util/compat.py +299 -0
- sqlalchemy/util/concurrency.py +322 -0
- sqlalchemy/util/cython.py +79 -0
- sqlalchemy/util/deprecations.py +401 -0
- sqlalchemy/util/langhelpers.py +2320 -0
- sqlalchemy/util/preloaded.py +152 -0
- sqlalchemy/util/queue.py +304 -0
- sqlalchemy/util/tool_support.py +201 -0
- sqlalchemy/util/topological.py +120 -0
- sqlalchemy/util/typing.py +711 -0
- sqlalchemy-2.1.0b2.dist-info/METADATA +269 -0
- sqlalchemy-2.1.0b2.dist-info/RECORD +270 -0
- sqlalchemy-2.1.0b2.dist-info/WHEEL +5 -0
- sqlalchemy-2.1.0b2.dist-info/licenses/LICENSE +19 -0
- sqlalchemy-2.1.0b2.dist-info/top_level.txt +1 -0
|
@@ -0,0 +1,408 @@
|
|
|
1
|
+
# dialects/postgresql/json.py
|
|
2
|
+
# Copyright (C) 2005-2026 the SQLAlchemy authors and contributors
|
|
3
|
+
# <see AUTHORS file>
|
|
4
|
+
#
|
|
5
|
+
# This module is part of SQLAlchemy and is released under
|
|
6
|
+
# the MIT License: https://www.opensource.org/licenses/mit-license.php
|
|
7
|
+
|
|
8
|
+
from __future__ import annotations
|
|
9
|
+
|
|
10
|
+
from typing import Any
|
|
11
|
+
from typing import Callable
|
|
12
|
+
from typing import List
|
|
13
|
+
from typing import Optional
|
|
14
|
+
from typing import TYPE_CHECKING
|
|
15
|
+
from typing import Union
|
|
16
|
+
|
|
17
|
+
from .array import ARRAY
|
|
18
|
+
from .array import array as _pg_array
|
|
19
|
+
from .operators import ASTEXT
|
|
20
|
+
from .operators import CONTAINED_BY
|
|
21
|
+
from .operators import CONTAINS
|
|
22
|
+
from .operators import DELETE_PATH
|
|
23
|
+
from .operators import HAS_ALL
|
|
24
|
+
from .operators import HAS_ANY
|
|
25
|
+
from .operators import HAS_KEY
|
|
26
|
+
from .operators import JSONPATH_ASTEXT
|
|
27
|
+
from .operators import PATH_EXISTS
|
|
28
|
+
from .operators import PATH_MATCH
|
|
29
|
+
from ... import types as sqltypes
|
|
30
|
+
from ...sql import cast
|
|
31
|
+
from ...sql.operators import OperatorClass
|
|
32
|
+
from ...sql.sqltypes import _CT_JSON
|
|
33
|
+
from ...sql.sqltypes import _T_JSON
|
|
34
|
+
|
|
35
|
+
if TYPE_CHECKING:
|
|
36
|
+
from ...engine.interfaces import Dialect
|
|
37
|
+
from ...sql.elements import ColumnElement
|
|
38
|
+
from ...sql.operators import OperatorType
|
|
39
|
+
from ...sql.type_api import _BindProcessorType
|
|
40
|
+
from ...sql.type_api import _LiteralProcessorType
|
|
41
|
+
from ...sql.type_api import TypeEngine
|
|
42
|
+
|
|
43
|
+
__all__ = ("JSON", "JSONB")
|
|
44
|
+
|
|
45
|
+
|
|
46
|
+
class JSONPathType(sqltypes.JSON.JSONPathType):
|
|
47
|
+
def _processor(
|
|
48
|
+
self, dialect: Dialect, super_proc: Optional[Callable[[Any], Any]]
|
|
49
|
+
) -> Callable[[Any], Any]:
|
|
50
|
+
def process(value: Any) -> Any:
|
|
51
|
+
if isinstance(value, str):
|
|
52
|
+
# If it's already a string assume that it's in json path
|
|
53
|
+
# format. This allows using cast with json paths literals
|
|
54
|
+
return value
|
|
55
|
+
elif value:
|
|
56
|
+
# If it's already a string assume that it's in json path
|
|
57
|
+
# format. This allows using cast with json paths literals
|
|
58
|
+
value = "{%s}" % (", ".join(map(str, value)))
|
|
59
|
+
else:
|
|
60
|
+
value = "{}"
|
|
61
|
+
if super_proc:
|
|
62
|
+
value = super_proc(value)
|
|
63
|
+
return value
|
|
64
|
+
|
|
65
|
+
return process
|
|
66
|
+
|
|
67
|
+
def bind_processor(self, dialect: Dialect) -> _BindProcessorType[Any]:
|
|
68
|
+
return self._processor(dialect, self.string_bind_processor(dialect)) # type: ignore[return-value] # noqa: E501
|
|
69
|
+
|
|
70
|
+
def literal_processor(
|
|
71
|
+
self, dialect: Dialect
|
|
72
|
+
) -> _LiteralProcessorType[Any]:
|
|
73
|
+
return self._processor(dialect, self.string_literal_processor(dialect)) # type: ignore[return-value] # noqa: E501
|
|
74
|
+
|
|
75
|
+
|
|
76
|
+
class JSONPATH(JSONPathType):
|
|
77
|
+
"""JSON Path Type.
|
|
78
|
+
|
|
79
|
+
This is usually required to cast literal values to json path when using
|
|
80
|
+
json search like function, such as ``jsonb_path_query_array`` or
|
|
81
|
+
``jsonb_path_exists``::
|
|
82
|
+
|
|
83
|
+
stmt = sa.select(
|
|
84
|
+
sa.func.jsonb_path_query_array(
|
|
85
|
+
table.c.jsonb_col, cast("$.address.id", JSONPATH)
|
|
86
|
+
)
|
|
87
|
+
)
|
|
88
|
+
|
|
89
|
+
"""
|
|
90
|
+
|
|
91
|
+
__visit_name__ = "JSONPATH"
|
|
92
|
+
|
|
93
|
+
|
|
94
|
+
class JSON(sqltypes.JSON[_T_JSON]):
|
|
95
|
+
"""Represent the PostgreSQL JSON type.
|
|
96
|
+
|
|
97
|
+
:class:`_postgresql.JSON` is used automatically whenever the base
|
|
98
|
+
:class:`_types.JSON` datatype is used against a PostgreSQL backend,
|
|
99
|
+
however base :class:`_types.JSON` datatype does not provide Python
|
|
100
|
+
accessors for PostgreSQL-specific comparison methods such as
|
|
101
|
+
:meth:`_postgresql.JSON.Comparator.astext`; additionally, to use
|
|
102
|
+
PostgreSQL ``JSONB``, the :class:`_postgresql.JSONB` datatype should
|
|
103
|
+
be used explicitly.
|
|
104
|
+
|
|
105
|
+
.. seealso::
|
|
106
|
+
|
|
107
|
+
:class:`_types.JSON` - main documentation for the generic
|
|
108
|
+
cross-platform JSON datatype.
|
|
109
|
+
|
|
110
|
+
The operators provided by the PostgreSQL version of :class:`_types.JSON`
|
|
111
|
+
include:
|
|
112
|
+
|
|
113
|
+
* Index operations (the ``->`` operator)::
|
|
114
|
+
|
|
115
|
+
data_table.c.data["some key"]
|
|
116
|
+
|
|
117
|
+
data_table.c.data[5]
|
|
118
|
+
|
|
119
|
+
* Index operations returning text
|
|
120
|
+
(the ``->>`` operator)::
|
|
121
|
+
|
|
122
|
+
data_table.c.data["some key"].astext == "some value"
|
|
123
|
+
|
|
124
|
+
Note that equivalent functionality is available via the
|
|
125
|
+
:attr:`.JSON.Comparator.as_string` accessor.
|
|
126
|
+
|
|
127
|
+
* Index operations with CAST
|
|
128
|
+
(equivalent to ``CAST(col ->> ['some key'] AS <type>)``)::
|
|
129
|
+
|
|
130
|
+
data_table.c.data["some key"].astext.cast(Integer) == 5
|
|
131
|
+
|
|
132
|
+
Note that equivalent functionality is available via the
|
|
133
|
+
:attr:`.JSON.Comparator.as_integer` and similar accessors.
|
|
134
|
+
|
|
135
|
+
* Path index operations (the ``#>`` operator)::
|
|
136
|
+
|
|
137
|
+
data_table.c.data[("key_1", "key_2", 5, ..., "key_n")]
|
|
138
|
+
|
|
139
|
+
* Path index operations returning text (the ``#>>`` operator)::
|
|
140
|
+
|
|
141
|
+
data_table.c.data[
|
|
142
|
+
("key_1", "key_2", 5, ..., "key_n")
|
|
143
|
+
].astext == "some value"
|
|
144
|
+
|
|
145
|
+
Index operations return an expression object whose type defaults to
|
|
146
|
+
:class:`_types.JSON` by default,
|
|
147
|
+
so that further JSON-oriented instructions
|
|
148
|
+
may be called upon the result type.
|
|
149
|
+
|
|
150
|
+
Custom serializers and deserializers are specified at the dialect level,
|
|
151
|
+
that is using :func:`_sa.create_engine`. The reason for this is that when
|
|
152
|
+
using psycopg2, the DBAPI only allows serializers at the per-cursor
|
|
153
|
+
or per-connection level. E.g.::
|
|
154
|
+
|
|
155
|
+
engine = create_engine(
|
|
156
|
+
"postgresql+psycopg2://scott:tiger@localhost/test",
|
|
157
|
+
json_serializer=my_serialize_fn,
|
|
158
|
+
json_deserializer=my_deserialize_fn,
|
|
159
|
+
)
|
|
160
|
+
|
|
161
|
+
When using the psycopg2 dialect, the json_deserializer is registered
|
|
162
|
+
against the database using ``psycopg2.extras.register_default_json``.
|
|
163
|
+
|
|
164
|
+
.. seealso::
|
|
165
|
+
|
|
166
|
+
:class:`_types.JSON` - Core level JSON type
|
|
167
|
+
|
|
168
|
+
:class:`_postgresql.JSONB`
|
|
169
|
+
|
|
170
|
+
""" # noqa
|
|
171
|
+
|
|
172
|
+
render_bind_cast = True
|
|
173
|
+
astext_type: TypeEngine[str] = sqltypes.Text()
|
|
174
|
+
|
|
175
|
+
def __init__(
|
|
176
|
+
self,
|
|
177
|
+
none_as_null: bool = False,
|
|
178
|
+
astext_type: Optional[TypeEngine[str]] = None,
|
|
179
|
+
):
|
|
180
|
+
"""Construct a :class:`_types.JSON` type.
|
|
181
|
+
|
|
182
|
+
:param none_as_null: if True, persist the value ``None`` as a
|
|
183
|
+
SQL NULL value, not the JSON encoding of ``null``. Note that
|
|
184
|
+
when this flag is False, the :func:`.null` construct can still
|
|
185
|
+
be used to persist a NULL value::
|
|
186
|
+
|
|
187
|
+
from sqlalchemy import null
|
|
188
|
+
|
|
189
|
+
conn.execute(table.insert(), {"data": null()})
|
|
190
|
+
|
|
191
|
+
.. seealso::
|
|
192
|
+
|
|
193
|
+
:attr:`_types.JSON.NULL`
|
|
194
|
+
|
|
195
|
+
:param astext_type: the type to use for the
|
|
196
|
+
:attr:`.JSON.Comparator.astext`
|
|
197
|
+
accessor on indexed attributes. Defaults to :class:`_types.Text`.
|
|
198
|
+
|
|
199
|
+
"""
|
|
200
|
+
super().__init__(none_as_null=none_as_null)
|
|
201
|
+
if astext_type is not None:
|
|
202
|
+
self.astext_type = astext_type
|
|
203
|
+
|
|
204
|
+
class Comparator(sqltypes.JSON.Comparator[_CT_JSON]):
|
|
205
|
+
"""Define comparison operations for :class:`_types.JSON`."""
|
|
206
|
+
|
|
207
|
+
type: JSON[_CT_JSON]
|
|
208
|
+
|
|
209
|
+
@property
|
|
210
|
+
def astext(self) -> ColumnElement[str]:
|
|
211
|
+
"""On an indexed expression, use the "astext" (e.g. "->>")
|
|
212
|
+
conversion when rendered in SQL.
|
|
213
|
+
|
|
214
|
+
E.g.::
|
|
215
|
+
|
|
216
|
+
select(data_table.c.data["some key"].astext)
|
|
217
|
+
|
|
218
|
+
.. seealso::
|
|
219
|
+
|
|
220
|
+
:meth:`_expression.ColumnElement.cast`
|
|
221
|
+
|
|
222
|
+
"""
|
|
223
|
+
if isinstance(self.expr.right.type, sqltypes.JSON.JSONPathType):
|
|
224
|
+
return self.expr.left.operate( # type: ignore[no-any-return]
|
|
225
|
+
JSONPATH_ASTEXT,
|
|
226
|
+
self.expr.right,
|
|
227
|
+
result_type=self.type.astext_type,
|
|
228
|
+
)
|
|
229
|
+
else:
|
|
230
|
+
return self.expr.left.operate( # type: ignore[no-any-return]
|
|
231
|
+
ASTEXT, self.expr.right, result_type=self.type.astext_type
|
|
232
|
+
)
|
|
233
|
+
|
|
234
|
+
comparator_factory = Comparator
|
|
235
|
+
|
|
236
|
+
|
|
237
|
+
class JSONB(JSON[_T_JSON]):
|
|
238
|
+
"""Represent the PostgreSQL JSONB type.
|
|
239
|
+
|
|
240
|
+
The :class:`_postgresql.JSONB` type stores arbitrary JSONB format data,
|
|
241
|
+
e.g.::
|
|
242
|
+
|
|
243
|
+
data_table = Table(
|
|
244
|
+
"data_table",
|
|
245
|
+
metadata,
|
|
246
|
+
Column("id", Integer, primary_key=True),
|
|
247
|
+
Column("data", JSONB),
|
|
248
|
+
)
|
|
249
|
+
|
|
250
|
+
with engine.connect() as conn:
|
|
251
|
+
conn.execute(
|
|
252
|
+
data_table.insert(), data={"key1": "value1", "key2": "value2"}
|
|
253
|
+
)
|
|
254
|
+
|
|
255
|
+
The :class:`_postgresql.JSONB` type includes all operations provided by
|
|
256
|
+
:class:`_types.JSON`, including the same behaviors for indexing
|
|
257
|
+
operations.
|
|
258
|
+
It also adds additional operators specific to JSONB, including
|
|
259
|
+
:meth:`.JSONB.Comparator.has_key`, :meth:`.JSONB.Comparator.has_all`,
|
|
260
|
+
:meth:`.JSONB.Comparator.has_any`, :meth:`.JSONB.Comparator.contains`,
|
|
261
|
+
:meth:`.JSONB.Comparator.contained_by`,
|
|
262
|
+
:meth:`.JSONB.Comparator.delete_path`,
|
|
263
|
+
:meth:`.JSONB.Comparator.path_exists` and
|
|
264
|
+
:meth:`.JSONB.Comparator.path_match`.
|
|
265
|
+
|
|
266
|
+
Like the :class:`_types.JSON` type, the :class:`_postgresql.JSONB`
|
|
267
|
+
type does not detect
|
|
268
|
+
in-place changes when used with the ORM, unless the
|
|
269
|
+
:mod:`sqlalchemy.ext.mutable` extension is used.
|
|
270
|
+
|
|
271
|
+
Custom serializers and deserializers
|
|
272
|
+
are shared with the :class:`_types.JSON` class,
|
|
273
|
+
using the ``json_serializer``
|
|
274
|
+
and ``json_deserializer`` keyword arguments. These must be specified
|
|
275
|
+
at the dialect level using :func:`_sa.create_engine`. When using
|
|
276
|
+
psycopg2, the serializers are associated with the jsonb type using
|
|
277
|
+
``psycopg2.extras.register_default_jsonb`` on a per-connection basis,
|
|
278
|
+
in the same way that ``psycopg2.extras.register_default_json`` is used
|
|
279
|
+
to register these handlers with the json type.
|
|
280
|
+
|
|
281
|
+
.. seealso::
|
|
282
|
+
|
|
283
|
+
:class:`_types.JSON`
|
|
284
|
+
|
|
285
|
+
.. warning::
|
|
286
|
+
|
|
287
|
+
**For applications that have indexes against JSONB subscript
|
|
288
|
+
expressions**
|
|
289
|
+
|
|
290
|
+
SQLAlchemy 2.0.42 made a change in how the subscript operation for
|
|
291
|
+
:class:`.JSONB` is rendered, from ``-> 'element'`` to ``['element']``,
|
|
292
|
+
for PostgreSQL versions greater than 14. This change caused an
|
|
293
|
+
unintended side effect for indexes that were created against
|
|
294
|
+
expressions that use subscript notation, e.g.
|
|
295
|
+
``Index("ix_entity_json_ab_text", data["a"]["b"].astext)``. If these
|
|
296
|
+
indexes were generated with the older syntax e.g. ``((entity.data ->
|
|
297
|
+
'a') ->> 'b')``, they will not be used by the PostgreSQL query planner
|
|
298
|
+
when a query is made using SQLAlchemy 2.0.42 or higher on PostgreSQL
|
|
299
|
+
versions 14 or higher. This occurs because the new text will resemble
|
|
300
|
+
``(entity.data['a'] ->> 'b')`` which will fail to produce the exact
|
|
301
|
+
textual syntax match required by the PostgreSQL query planner.
|
|
302
|
+
Therefore, for users upgrading to SQLAlchemy 2.0.42 or higher, existing
|
|
303
|
+
indexes that were created against :class:`.JSONB` expressions that use
|
|
304
|
+
subscripting would need to be dropped and re-created in order for them
|
|
305
|
+
to work with the new query syntax, e.g. an expression like
|
|
306
|
+
``((entity.data -> 'a') ->> 'b')`` would become ``(entity.data['a'] ->>
|
|
307
|
+
'b')``.
|
|
308
|
+
|
|
309
|
+
.. seealso::
|
|
310
|
+
|
|
311
|
+
:ticket:`12868` - discussion of this issue
|
|
312
|
+
|
|
313
|
+
"""
|
|
314
|
+
|
|
315
|
+
__visit_name__ = "JSONB"
|
|
316
|
+
|
|
317
|
+
operator_classes = OperatorClass.JSON | OperatorClass.CONCATENABLE
|
|
318
|
+
|
|
319
|
+
def coerce_compared_value(
|
|
320
|
+
self, op: Optional[OperatorType], value: Any
|
|
321
|
+
) -> TypeEngine[Any]:
|
|
322
|
+
if op in (PATH_MATCH, PATH_EXISTS):
|
|
323
|
+
return JSON.JSONPathType()
|
|
324
|
+
else:
|
|
325
|
+
return super().coerce_compared_value(op, value)
|
|
326
|
+
|
|
327
|
+
class Comparator(JSON.Comparator[_CT_JSON]):
|
|
328
|
+
"""Define comparison operations for :class:`_types.JSON`."""
|
|
329
|
+
|
|
330
|
+
type: JSONB[_CT_JSON]
|
|
331
|
+
|
|
332
|
+
def has_key(self, other: Any) -> ColumnElement[bool]:
|
|
333
|
+
"""Boolean expression. Test for presence of a key (equivalent of
|
|
334
|
+
the ``?`` operator). Note that the key may be a SQLA expression.
|
|
335
|
+
"""
|
|
336
|
+
return self.operate(HAS_KEY, other, result_type=sqltypes.Boolean)
|
|
337
|
+
|
|
338
|
+
def has_all(self, other: Any) -> ColumnElement[bool]:
|
|
339
|
+
"""Boolean expression. Test for presence of all keys in jsonb
|
|
340
|
+
(equivalent of the ``?&`` operator)
|
|
341
|
+
"""
|
|
342
|
+
return self.operate(HAS_ALL, other, result_type=sqltypes.Boolean)
|
|
343
|
+
|
|
344
|
+
def has_any(self, other: Any) -> ColumnElement[bool]:
|
|
345
|
+
"""Boolean expression. Test for presence of any key in jsonb
|
|
346
|
+
(equivalent of the ``?|`` operator)
|
|
347
|
+
"""
|
|
348
|
+
return self.operate(HAS_ANY, other, result_type=sqltypes.Boolean)
|
|
349
|
+
|
|
350
|
+
def contains(self, other: Any, **kwargs: Any) -> ColumnElement[bool]:
|
|
351
|
+
"""Boolean expression. Test if keys (or array) are a superset
|
|
352
|
+
of/contained the keys of the argument jsonb expression
|
|
353
|
+
(equivalent of the ``@>`` operator).
|
|
354
|
+
|
|
355
|
+
kwargs may be ignored by this operator but are required for API
|
|
356
|
+
conformance.
|
|
357
|
+
"""
|
|
358
|
+
return self.operate(CONTAINS, other, result_type=sqltypes.Boolean)
|
|
359
|
+
|
|
360
|
+
def contained_by(self, other: Any) -> ColumnElement[bool]:
|
|
361
|
+
"""Boolean expression. Test if keys are a proper subset of the
|
|
362
|
+
keys of the argument jsonb expression
|
|
363
|
+
(equivalent of the ``<@`` operator).
|
|
364
|
+
"""
|
|
365
|
+
return self.operate(
|
|
366
|
+
CONTAINED_BY, other, result_type=sqltypes.Boolean
|
|
367
|
+
)
|
|
368
|
+
|
|
369
|
+
def delete_path(
|
|
370
|
+
self, array: Union[List[str], _pg_array[str]]
|
|
371
|
+
) -> ColumnElement[_CT_JSON]:
|
|
372
|
+
"""JSONB expression. Deletes field or array element specified in
|
|
373
|
+
the argument array (equivalent of the ``#-`` operator).
|
|
374
|
+
|
|
375
|
+
The input may be a list of strings that will be coerced to an
|
|
376
|
+
``ARRAY`` or an instance of :meth:`_postgres.array`.
|
|
377
|
+
|
|
378
|
+
.. versionadded:: 2.0
|
|
379
|
+
"""
|
|
380
|
+
if not isinstance(array, _pg_array):
|
|
381
|
+
array = _pg_array(array)
|
|
382
|
+
right_side = cast(array, ARRAY(sqltypes.TEXT))
|
|
383
|
+
return self.operate(DELETE_PATH, right_side, result_type=JSONB)
|
|
384
|
+
|
|
385
|
+
def path_exists(self, other: Any) -> ColumnElement[bool]:
|
|
386
|
+
"""Boolean expression. Test for presence of item given by the
|
|
387
|
+
argument JSONPath expression (equivalent of the ``@?`` operator).
|
|
388
|
+
|
|
389
|
+
.. versionadded:: 2.0
|
|
390
|
+
"""
|
|
391
|
+
return self.operate(
|
|
392
|
+
PATH_EXISTS, other, result_type=sqltypes.Boolean
|
|
393
|
+
)
|
|
394
|
+
|
|
395
|
+
def path_match(self, other: Any) -> ColumnElement[bool]:
|
|
396
|
+
"""Boolean expression. Test if JSONPath predicate given by the
|
|
397
|
+
argument JSONPath expression matches
|
|
398
|
+
(equivalent of the ``@@`` operator).
|
|
399
|
+
|
|
400
|
+
Only the first item of the result is taken into account.
|
|
401
|
+
|
|
402
|
+
.. versionadded:: 2.0
|
|
403
|
+
"""
|
|
404
|
+
return self.operate(
|
|
405
|
+
PATH_MATCH, other, result_type=sqltypes.Boolean
|
|
406
|
+
)
|
|
407
|
+
|
|
408
|
+
comparator_factory = Comparator
|