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.
Files changed (270) hide show
  1. sqlalchemy/__init__.py +298 -0
  2. sqlalchemy/connectors/__init__.py +18 -0
  3. sqlalchemy/connectors/aioodbc.py +171 -0
  4. sqlalchemy/connectors/asyncio.py +476 -0
  5. sqlalchemy/connectors/pyodbc.py +250 -0
  6. sqlalchemy/dialects/__init__.py +62 -0
  7. sqlalchemy/dialects/_typing.py +30 -0
  8. sqlalchemy/dialects/mssql/__init__.py +89 -0
  9. sqlalchemy/dialects/mssql/aioodbc.py +63 -0
  10. sqlalchemy/dialects/mssql/base.py +4166 -0
  11. sqlalchemy/dialects/mssql/information_schema.py +285 -0
  12. sqlalchemy/dialects/mssql/json.py +140 -0
  13. sqlalchemy/dialects/mssql/mssqlpython.py +220 -0
  14. sqlalchemy/dialects/mssql/provision.py +196 -0
  15. sqlalchemy/dialects/mssql/pymssql.py +126 -0
  16. sqlalchemy/dialects/mssql/pyodbc.py +698 -0
  17. sqlalchemy/dialects/mysql/__init__.py +106 -0
  18. sqlalchemy/dialects/mysql/_mariadb_shim.py +312 -0
  19. sqlalchemy/dialects/mysql/aiomysql.py +226 -0
  20. sqlalchemy/dialects/mysql/asyncmy.py +214 -0
  21. sqlalchemy/dialects/mysql/base.py +3877 -0
  22. sqlalchemy/dialects/mysql/cymysql.py +106 -0
  23. sqlalchemy/dialects/mysql/dml.py +279 -0
  24. sqlalchemy/dialects/mysql/enumerated.py +277 -0
  25. sqlalchemy/dialects/mysql/expression.py +146 -0
  26. sqlalchemy/dialects/mysql/json.py +92 -0
  27. sqlalchemy/dialects/mysql/mariadb.py +67 -0
  28. sqlalchemy/dialects/mysql/mariadbconnector.py +330 -0
  29. sqlalchemy/dialects/mysql/mysqlconnector.py +296 -0
  30. sqlalchemy/dialects/mysql/mysqldb.py +312 -0
  31. sqlalchemy/dialects/mysql/provision.py +153 -0
  32. sqlalchemy/dialects/mysql/pymysql.py +157 -0
  33. sqlalchemy/dialects/mysql/pyodbc.py +156 -0
  34. sqlalchemy/dialects/mysql/reflection.py +724 -0
  35. sqlalchemy/dialects/mysql/reserved_words.py +570 -0
  36. sqlalchemy/dialects/mysql/types.py +845 -0
  37. sqlalchemy/dialects/oracle/__init__.py +85 -0
  38. sqlalchemy/dialects/oracle/base.py +3977 -0
  39. sqlalchemy/dialects/oracle/cx_oracle.py +1601 -0
  40. sqlalchemy/dialects/oracle/dictionary.py +507 -0
  41. sqlalchemy/dialects/oracle/json.py +158 -0
  42. sqlalchemy/dialects/oracle/oracledb.py +909 -0
  43. sqlalchemy/dialects/oracle/provision.py +288 -0
  44. sqlalchemy/dialects/oracle/types.py +367 -0
  45. sqlalchemy/dialects/oracle/vector.py +368 -0
  46. sqlalchemy/dialects/postgresql/__init__.py +171 -0
  47. sqlalchemy/dialects/postgresql/_psycopg_common.py +229 -0
  48. sqlalchemy/dialects/postgresql/array.py +534 -0
  49. sqlalchemy/dialects/postgresql/asyncpg.py +1323 -0
  50. sqlalchemy/dialects/postgresql/base.py +5789 -0
  51. sqlalchemy/dialects/postgresql/bitstring.py +327 -0
  52. sqlalchemy/dialects/postgresql/dml.py +360 -0
  53. sqlalchemy/dialects/postgresql/ext.py +593 -0
  54. sqlalchemy/dialects/postgresql/hstore.py +423 -0
  55. sqlalchemy/dialects/postgresql/json.py +408 -0
  56. sqlalchemy/dialects/postgresql/named_types.py +521 -0
  57. sqlalchemy/dialects/postgresql/operators.py +130 -0
  58. sqlalchemy/dialects/postgresql/pg8000.py +670 -0
  59. sqlalchemy/dialects/postgresql/pg_catalog.py +344 -0
  60. sqlalchemy/dialects/postgresql/provision.py +184 -0
  61. sqlalchemy/dialects/postgresql/psycopg.py +799 -0
  62. sqlalchemy/dialects/postgresql/psycopg2.py +860 -0
  63. sqlalchemy/dialects/postgresql/psycopg2cffi.py +61 -0
  64. sqlalchemy/dialects/postgresql/ranges.py +1002 -0
  65. sqlalchemy/dialects/postgresql/types.py +388 -0
  66. sqlalchemy/dialects/sqlite/__init__.py +57 -0
  67. sqlalchemy/dialects/sqlite/aiosqlite.py +321 -0
  68. sqlalchemy/dialects/sqlite/base.py +3063 -0
  69. sqlalchemy/dialects/sqlite/dml.py +279 -0
  70. sqlalchemy/dialects/sqlite/json.py +100 -0
  71. sqlalchemy/dialects/sqlite/provision.py +229 -0
  72. sqlalchemy/dialects/sqlite/pysqlcipher.py +161 -0
  73. sqlalchemy/dialects/sqlite/pysqlite.py +754 -0
  74. sqlalchemy/dialects/type_migration_guidelines.txt +145 -0
  75. sqlalchemy/engine/__init__.py +62 -0
  76. sqlalchemy/engine/_processors_cy.cp313t-win_arm64.pyd +0 -0
  77. sqlalchemy/engine/_processors_cy.py +92 -0
  78. sqlalchemy/engine/_result_cy.cp313t-win_arm64.pyd +0 -0
  79. sqlalchemy/engine/_result_cy.py +633 -0
  80. sqlalchemy/engine/_row_cy.cp313t-win_arm64.pyd +0 -0
  81. sqlalchemy/engine/_row_cy.py +232 -0
  82. sqlalchemy/engine/_util_cy.cp313t-win_arm64.pyd +0 -0
  83. sqlalchemy/engine/_util_cy.py +136 -0
  84. sqlalchemy/engine/base.py +3354 -0
  85. sqlalchemy/engine/characteristics.py +155 -0
  86. sqlalchemy/engine/create.py +877 -0
  87. sqlalchemy/engine/cursor.py +2421 -0
  88. sqlalchemy/engine/default.py +2402 -0
  89. sqlalchemy/engine/events.py +965 -0
  90. sqlalchemy/engine/interfaces.py +3495 -0
  91. sqlalchemy/engine/mock.py +134 -0
  92. sqlalchemy/engine/processors.py +82 -0
  93. sqlalchemy/engine/reflection.py +2100 -0
  94. sqlalchemy/engine/result.py +1966 -0
  95. sqlalchemy/engine/row.py +397 -0
  96. sqlalchemy/engine/strategies.py +16 -0
  97. sqlalchemy/engine/url.py +922 -0
  98. sqlalchemy/engine/util.py +156 -0
  99. sqlalchemy/event/__init__.py +26 -0
  100. sqlalchemy/event/api.py +220 -0
  101. sqlalchemy/event/attr.py +674 -0
  102. sqlalchemy/event/base.py +472 -0
  103. sqlalchemy/event/legacy.py +258 -0
  104. sqlalchemy/event/registry.py +390 -0
  105. sqlalchemy/events.py +17 -0
  106. sqlalchemy/exc.py +922 -0
  107. sqlalchemy/ext/__init__.py +11 -0
  108. sqlalchemy/ext/associationproxy.py +2072 -0
  109. sqlalchemy/ext/asyncio/__init__.py +29 -0
  110. sqlalchemy/ext/asyncio/base.py +281 -0
  111. sqlalchemy/ext/asyncio/engine.py +1487 -0
  112. sqlalchemy/ext/asyncio/exc.py +21 -0
  113. sqlalchemy/ext/asyncio/result.py +994 -0
  114. sqlalchemy/ext/asyncio/scoping.py +1679 -0
  115. sqlalchemy/ext/asyncio/session.py +2007 -0
  116. sqlalchemy/ext/automap.py +1701 -0
  117. sqlalchemy/ext/baked.py +559 -0
  118. sqlalchemy/ext/compiler.py +600 -0
  119. sqlalchemy/ext/declarative/__init__.py +65 -0
  120. sqlalchemy/ext/declarative/extensions.py +560 -0
  121. sqlalchemy/ext/horizontal_shard.py +481 -0
  122. sqlalchemy/ext/hybrid.py +1877 -0
  123. sqlalchemy/ext/indexable.py +364 -0
  124. sqlalchemy/ext/instrumentation.py +450 -0
  125. sqlalchemy/ext/mutable.py +1081 -0
  126. sqlalchemy/ext/orderinglist.py +439 -0
  127. sqlalchemy/ext/serializer.py +185 -0
  128. sqlalchemy/future/__init__.py +16 -0
  129. sqlalchemy/future/engine.py +15 -0
  130. sqlalchemy/inspection.py +174 -0
  131. sqlalchemy/log.py +283 -0
  132. sqlalchemy/orm/__init__.py +176 -0
  133. sqlalchemy/orm/_orm_constructors.py +2694 -0
  134. sqlalchemy/orm/_typing.py +179 -0
  135. sqlalchemy/orm/attributes.py +2868 -0
  136. sqlalchemy/orm/base.py +976 -0
  137. sqlalchemy/orm/bulk_persistence.py +2152 -0
  138. sqlalchemy/orm/clsregistry.py +582 -0
  139. sqlalchemy/orm/collections.py +1568 -0
  140. sqlalchemy/orm/context.py +3471 -0
  141. sqlalchemy/orm/decl_api.py +2280 -0
  142. sqlalchemy/orm/decl_base.py +2309 -0
  143. sqlalchemy/orm/dependency.py +1306 -0
  144. sqlalchemy/orm/descriptor_props.py +1183 -0
  145. sqlalchemy/orm/dynamic.py +307 -0
  146. sqlalchemy/orm/evaluator.py +379 -0
  147. sqlalchemy/orm/events.py +3386 -0
  148. sqlalchemy/orm/exc.py +237 -0
  149. sqlalchemy/orm/identity.py +302 -0
  150. sqlalchemy/orm/instrumentation.py +746 -0
  151. sqlalchemy/orm/interfaces.py +1589 -0
  152. sqlalchemy/orm/loading.py +1684 -0
  153. sqlalchemy/orm/mapped_collection.py +557 -0
  154. sqlalchemy/orm/mapper.py +4411 -0
  155. sqlalchemy/orm/path_registry.py +829 -0
  156. sqlalchemy/orm/persistence.py +1789 -0
  157. sqlalchemy/orm/properties.py +973 -0
  158. sqlalchemy/orm/query.py +3528 -0
  159. sqlalchemy/orm/relationships.py +3570 -0
  160. sqlalchemy/orm/scoping.py +2232 -0
  161. sqlalchemy/orm/session.py +5403 -0
  162. sqlalchemy/orm/state.py +1175 -0
  163. sqlalchemy/orm/state_changes.py +196 -0
  164. sqlalchemy/orm/strategies.py +3492 -0
  165. sqlalchemy/orm/strategy_options.py +2562 -0
  166. sqlalchemy/orm/sync.py +164 -0
  167. sqlalchemy/orm/unitofwork.py +798 -0
  168. sqlalchemy/orm/util.py +2438 -0
  169. sqlalchemy/orm/writeonly.py +694 -0
  170. sqlalchemy/pool/__init__.py +41 -0
  171. sqlalchemy/pool/base.py +1522 -0
  172. sqlalchemy/pool/events.py +375 -0
  173. sqlalchemy/pool/impl.py +582 -0
  174. sqlalchemy/py.typed +0 -0
  175. sqlalchemy/schema.py +74 -0
  176. sqlalchemy/sql/__init__.py +156 -0
  177. sqlalchemy/sql/_annotated_cols.py +397 -0
  178. sqlalchemy/sql/_dml_constructors.py +132 -0
  179. sqlalchemy/sql/_elements_constructors.py +2164 -0
  180. sqlalchemy/sql/_orm_types.py +20 -0
  181. sqlalchemy/sql/_selectable_constructors.py +840 -0
  182. sqlalchemy/sql/_typing.py +487 -0
  183. sqlalchemy/sql/_util_cy.cp313t-win_arm64.pyd +0 -0
  184. sqlalchemy/sql/_util_cy.py +127 -0
  185. sqlalchemy/sql/annotation.py +590 -0
  186. sqlalchemy/sql/base.py +2699 -0
  187. sqlalchemy/sql/cache_key.py +1066 -0
  188. sqlalchemy/sql/coercions.py +1373 -0
  189. sqlalchemy/sql/compiler.py +8327 -0
  190. sqlalchemy/sql/crud.py +1815 -0
  191. sqlalchemy/sql/ddl.py +1928 -0
  192. sqlalchemy/sql/default_comparator.py +654 -0
  193. sqlalchemy/sql/dml.py +1977 -0
  194. sqlalchemy/sql/elements.py +6033 -0
  195. sqlalchemy/sql/events.py +458 -0
  196. sqlalchemy/sql/expression.py +172 -0
  197. sqlalchemy/sql/functions.py +2305 -0
  198. sqlalchemy/sql/lambdas.py +1443 -0
  199. sqlalchemy/sql/naming.py +209 -0
  200. sqlalchemy/sql/operators.py +2897 -0
  201. sqlalchemy/sql/roles.py +332 -0
  202. sqlalchemy/sql/schema.py +6703 -0
  203. sqlalchemy/sql/selectable.py +7553 -0
  204. sqlalchemy/sql/sqltypes.py +4093 -0
  205. sqlalchemy/sql/traversals.py +1042 -0
  206. sqlalchemy/sql/type_api.py +2446 -0
  207. sqlalchemy/sql/util.py +1495 -0
  208. sqlalchemy/sql/visitors.py +1157 -0
  209. sqlalchemy/testing/__init__.py +96 -0
  210. sqlalchemy/testing/assertions.py +1007 -0
  211. sqlalchemy/testing/assertsql.py +519 -0
  212. sqlalchemy/testing/asyncio.py +128 -0
  213. sqlalchemy/testing/config.py +440 -0
  214. sqlalchemy/testing/engines.py +483 -0
  215. sqlalchemy/testing/entities.py +117 -0
  216. sqlalchemy/testing/exclusions.py +476 -0
  217. sqlalchemy/testing/fixtures/__init__.py +30 -0
  218. sqlalchemy/testing/fixtures/base.py +384 -0
  219. sqlalchemy/testing/fixtures/mypy.py +247 -0
  220. sqlalchemy/testing/fixtures/orm.py +227 -0
  221. sqlalchemy/testing/fixtures/sql.py +538 -0
  222. sqlalchemy/testing/pickleable.py +155 -0
  223. sqlalchemy/testing/plugin/__init__.py +6 -0
  224. sqlalchemy/testing/plugin/bootstrap.py +51 -0
  225. sqlalchemy/testing/plugin/plugin_base.py +828 -0
  226. sqlalchemy/testing/plugin/pytestplugin.py +892 -0
  227. sqlalchemy/testing/profiling.py +329 -0
  228. sqlalchemy/testing/provision.py +613 -0
  229. sqlalchemy/testing/requirements.py +1978 -0
  230. sqlalchemy/testing/schema.py +198 -0
  231. sqlalchemy/testing/suite/__init__.py +19 -0
  232. sqlalchemy/testing/suite/test_cte.py +237 -0
  233. sqlalchemy/testing/suite/test_ddl.py +420 -0
  234. sqlalchemy/testing/suite/test_dialect.py +776 -0
  235. sqlalchemy/testing/suite/test_insert.py +630 -0
  236. sqlalchemy/testing/suite/test_reflection.py +3557 -0
  237. sqlalchemy/testing/suite/test_results.py +660 -0
  238. sqlalchemy/testing/suite/test_rowcount.py +258 -0
  239. sqlalchemy/testing/suite/test_select.py +2112 -0
  240. sqlalchemy/testing/suite/test_sequence.py +317 -0
  241. sqlalchemy/testing/suite/test_table_via_select.py +686 -0
  242. sqlalchemy/testing/suite/test_types.py +2271 -0
  243. sqlalchemy/testing/suite/test_unicode_ddl.py +189 -0
  244. sqlalchemy/testing/suite/test_update_delete.py +139 -0
  245. sqlalchemy/testing/util.py +535 -0
  246. sqlalchemy/testing/warnings.py +52 -0
  247. sqlalchemy/types.py +76 -0
  248. sqlalchemy/util/__init__.py +158 -0
  249. sqlalchemy/util/_collections.py +688 -0
  250. sqlalchemy/util/_collections_cy.cp313t-win_arm64.pyd +0 -0
  251. sqlalchemy/util/_collections_cy.pxd +8 -0
  252. sqlalchemy/util/_collections_cy.py +516 -0
  253. sqlalchemy/util/_has_cython.py +46 -0
  254. sqlalchemy/util/_immutabledict_cy.cp313t-win_arm64.pyd +0 -0
  255. sqlalchemy/util/_immutabledict_cy.py +240 -0
  256. sqlalchemy/util/compat.py +299 -0
  257. sqlalchemy/util/concurrency.py +322 -0
  258. sqlalchemy/util/cython.py +79 -0
  259. sqlalchemy/util/deprecations.py +401 -0
  260. sqlalchemy/util/langhelpers.py +2320 -0
  261. sqlalchemy/util/preloaded.py +152 -0
  262. sqlalchemy/util/queue.py +304 -0
  263. sqlalchemy/util/tool_support.py +201 -0
  264. sqlalchemy/util/topological.py +120 -0
  265. sqlalchemy/util/typing.py +711 -0
  266. sqlalchemy-2.1.0b2.dist-info/METADATA +269 -0
  267. sqlalchemy-2.1.0b2.dist-info/RECORD +270 -0
  268. sqlalchemy-2.1.0b2.dist-info/WHEEL +5 -0
  269. sqlalchemy-2.1.0b2.dist-info/licenses/LICENSE +19 -0
  270. 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