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,521 @@
1
+ # dialects/postgresql/named_types.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
+ # mypy: ignore-errors
8
+ from __future__ import annotations
9
+
10
+ from types import ModuleType
11
+ from typing import Any
12
+ from typing import Dict
13
+ from typing import Optional
14
+ from typing import Type
15
+ from typing import TYPE_CHECKING
16
+ from typing import Union
17
+
18
+ from ... import schema
19
+ from ... import util
20
+ from ...sql import coercions
21
+ from ...sql import elements
22
+ from ...sql import roles
23
+ from ...sql import sqltypes
24
+ from ...sql import type_api
25
+ from ...sql.base import _NoArg
26
+ from ...sql.ddl import CheckFirst
27
+ from ...sql.ddl import InvokeCreateDDLBase
28
+ from ...sql.ddl import InvokeDropDDLBase
29
+
30
+ if TYPE_CHECKING:
31
+ from ...sql._typing import _CreateDropBind
32
+ from ...sql._typing import _TypeEngineArgument
33
+
34
+
35
+ class NamedType(schema.SchemaVisitable, sqltypes.TypeEngine):
36
+ """Base for named types."""
37
+
38
+ __abstract__ = True
39
+ DDLGenerator: Type[NamedTypeGenerator]
40
+ DDLDropper: Type[NamedTypeDropper]
41
+ create_type: bool
42
+
43
+ def create(
44
+ self, bind: _CreateDropBind, checkfirst: bool = True, **kw: Any
45
+ ) -> None:
46
+ """Emit ``CREATE`` DDL for this type.
47
+
48
+ :param bind: a connectable :class:`_engine.Engine`,
49
+ :class:`_engine.Connection`, or similar object to emit
50
+ SQL.
51
+ :param checkfirst: if ``True``, a query against
52
+ the PG catalog will be first performed to see
53
+ if the type does not exist already before
54
+ creating.
55
+
56
+ """
57
+ bind._run_ddl_visitor(self.DDLGenerator, self, checkfirst=checkfirst)
58
+
59
+ def drop(
60
+ self, bind: _CreateDropBind, checkfirst: bool = True, **kw: Any
61
+ ) -> None:
62
+ """Emit ``DROP`` DDL for this type.
63
+
64
+ :param bind: a connectable :class:`_engine.Engine`,
65
+ :class:`_engine.Connection`, or similar object to emit
66
+ SQL.
67
+ :param checkfirst: if ``True``, a query against
68
+ the PG catalog will be first performed to see
69
+ if the type actually exists before dropping.
70
+
71
+ """
72
+ bind._run_ddl_visitor(self.DDLDropper, self, checkfirst=checkfirst)
73
+
74
+ def _check_for_name_in_memos(
75
+ self, checkfirst: CheckFirst, kw: Dict[str, Any]
76
+ ) -> bool:
77
+ """Look in the 'ddl runner' for 'memos', then
78
+ note our name in that collection.
79
+
80
+ This to ensure a particular named type is operated
81
+ upon only once within any kind of create/drop
82
+ sequence without relying upon "checkfirst".
83
+
84
+ """
85
+ if not self.create_type:
86
+ return True
87
+ if "_ddl_runner" in kw:
88
+ ddl_runner = kw["_ddl_runner"]
89
+ type_name = f"pg_{self.__visit_name__}"
90
+ if type_name in ddl_runner.memo:
91
+ existing = ddl_runner.memo[type_name]
92
+ else:
93
+ existing = ddl_runner.memo[type_name] = set()
94
+ present = (self.schema, self.name) in existing
95
+ existing.add((self.schema, self.name))
96
+ return present
97
+ else:
98
+ return False
99
+
100
+ def _on_table_create(
101
+ self,
102
+ target: schema.Table,
103
+ bind: _CreateDropBind,
104
+ checkfirst: Union[bool, CheckFirst] = CheckFirst.NONE,
105
+ **kw: Any,
106
+ ) -> None:
107
+ checkfirst = CheckFirst(checkfirst) & CheckFirst.TYPES
108
+ if not self._check_for_name_in_memos(checkfirst, kw):
109
+ self.create(bind=bind, checkfirst=bool(checkfirst))
110
+
111
+ def _on_table_drop(
112
+ self,
113
+ target: Any,
114
+ bind: _CreateDropBind,
115
+ checkfirst: CheckFirst = CheckFirst.NONE,
116
+ **kw: Any,
117
+ ) -> None:
118
+ # do nothing since the enum is attached to a metadata
119
+ assert self.metadata is not None
120
+
121
+ def _on_metadata_create(
122
+ self,
123
+ target: schema.MetaData,
124
+ bind: _CreateDropBind,
125
+ checkfirst: Union[bool, CheckFirst] = CheckFirst.NONE,
126
+ **kw: Any,
127
+ ) -> None:
128
+ checkfirst = CheckFirst(checkfirst) & CheckFirst.TYPES
129
+ if not self._check_for_name_in_memos(checkfirst, kw):
130
+ self.create(bind=bind, checkfirst=bool(checkfirst))
131
+
132
+ def _on_metadata_drop(
133
+ self,
134
+ target: schema.MetaData,
135
+ bind: _CreateDropBind,
136
+ checkfirst: Union[bool, CheckFirst] = CheckFirst.NONE,
137
+ **kw: Any,
138
+ ) -> None:
139
+ checkfirst = CheckFirst(checkfirst) & CheckFirst.TYPES
140
+ if not self._check_for_name_in_memos(checkfirst, kw):
141
+ self.drop(bind=bind, checkfirst=bool(checkfirst))
142
+
143
+
144
+ class NamedTypeGenerator(InvokeCreateDDLBase):
145
+ def __init__(self, dialect, connection, checkfirst=False, **kwargs):
146
+ super().__init__(connection, **kwargs)
147
+ self.checkfirst = checkfirst
148
+
149
+ def _can_create_type(self, type_):
150
+ if not self.checkfirst:
151
+ return True
152
+
153
+ effective_schema = self.connection.schema_for_object(type_)
154
+ return not self.connection.dialect.has_type(
155
+ self.connection, type_.name, schema=effective_schema
156
+ )
157
+
158
+
159
+ class NamedTypeDropper(InvokeDropDDLBase):
160
+ def __init__(self, dialect, connection, checkfirst=False, **kwargs):
161
+ super().__init__(connection, **kwargs)
162
+ self.checkfirst = checkfirst
163
+
164
+ def _can_drop_type(self, type_):
165
+ if not self.checkfirst:
166
+ return True
167
+
168
+ effective_schema = self.connection.schema_for_object(type_)
169
+ return self.connection.dialect.has_type(
170
+ self.connection, type_.name, schema=effective_schema
171
+ )
172
+
173
+
174
+ class EnumGenerator(NamedTypeGenerator):
175
+ def visit_enum(self, enum):
176
+ if not self._can_create_type(enum):
177
+ return
178
+
179
+ with self.with_ddl_events(enum):
180
+ self.connection.execute(CreateEnumType(enum))
181
+
182
+
183
+ class EnumDropper(NamedTypeDropper):
184
+ def visit_enum(self, enum):
185
+ if not self._can_drop_type(enum):
186
+ return
187
+
188
+ with self.with_ddl_events(enum):
189
+ self.connection.execute(DropEnumType(enum))
190
+
191
+
192
+ class ENUM(NamedType, type_api.NativeForEmulated, sqltypes.Enum):
193
+ """PostgreSQL ENUM type.
194
+
195
+ This is a subclass of :class:`_types.Enum` which includes
196
+ support for PG's ``CREATE TYPE`` and ``DROP TYPE``.
197
+
198
+ When the builtin type :class:`_types.Enum` is used and the
199
+ :paramref:`.Enum.native_enum` flag is left at its default of
200
+ True, the PostgreSQL backend will use a :class:`_postgresql.ENUM`
201
+ type as the implementation, so the special create/drop rules
202
+ will be used.
203
+
204
+ The create/drop behavior of ENUM tries to follow the PostgreSQL behavior,
205
+ with an usability improvement indicated below.
206
+
207
+ When using :class:`_types.Enum` or :class:`_postgresql.ENUM`
208
+ in an "inline" fashion, the ``CREATE TYPE`` is emitted
209
+ corresponding to when the :meth:`_schema.Table.create` method is called::
210
+
211
+ table = Table(
212
+ "sometable",
213
+ metadata,
214
+ Column("some_enum", ENUM("a", "b", "c", name="myenum")),
215
+ )
216
+
217
+ # will check if enum exists and emit CREATE ENUM then CREATE TABLE
218
+ table.create(engine)
219
+ table.drop(engine) # will *not* drop the enum.
220
+
221
+ The enum will not be dropped when the table is dropped, since it's
222
+ associated with the metadata, not the table itself. Call drop on the
223
+ :class:`_postgresql.ENUM` directly to drop the type::
224
+
225
+ metadata.get_schema_object_by_name("enum", "myenum").drop(engine)
226
+
227
+ To use a common enumerated type between multiple tables, the best
228
+ practice is to declare the :class:`_types.Enum` or
229
+ :class:`_postgresql.ENUM` independently::
230
+
231
+ my_enum = ENUM("a", "b", "c", name="myenum", metadata=metadata)
232
+
233
+ t1 = Table("sometable_one", metadata, Column("some_enum", myenum))
234
+
235
+ t2 = Table("sometable_two", metadata, Column("some_enum", myenum))
236
+
237
+ Like before, the type will be created if it does not exist::
238
+
239
+ # will check if enum exists and emit CREATE ENUM then CREATE TABLE
240
+ t1.create(engine)
241
+
242
+ The type will always be created and dropped if either the metadata-wide
243
+ create/drop is called::
244
+
245
+ metadata.create_all(engine) # will emit CREATE TYPE
246
+ metadata.drop_all(engine) # will emit DROP TYPE
247
+
248
+ The type can also be created and dropped directly::
249
+
250
+ my_enum.create(engine)
251
+ my_enum.drop(engine)
252
+
253
+ .. versionchanged:: 2.1 The behavior of :class:`_postgresql.ENUM` and
254
+ other named types has been changed to better reflect how PostgreSQL
255
+ handles CREATE TYPE and DROP TYPE operations.
256
+ Named types are still created when needed during table
257
+ creation if they do not already exist. However, they are no longer
258
+ dropped for an individual :meth:`.Table.drop` operation, since the type
259
+ may be referenced by other tables as well. Instead,
260
+ :meth:`.Enum.drop` may be used or :meth:`.MetaData.drop_all` will drop
261
+ all associated types.
262
+
263
+ """
264
+
265
+ native_enum = True
266
+ DDLGenerator = EnumGenerator
267
+ DDLDropper = EnumDropper
268
+
269
+ def __init__(
270
+ self,
271
+ *enums,
272
+ name: Union[str, _NoArg, None] = _NoArg.NO_ARG,
273
+ create_type: bool = True,
274
+ **kw,
275
+ ):
276
+ """Construct an :class:`_postgresql.ENUM`.
277
+
278
+ Arguments are the same as that of
279
+ :class:`_types.Enum`, but also including
280
+ the following parameters.
281
+
282
+ :param create_type: Defaults to True.
283
+ Indicates that ``CREATE TYPE`` should be
284
+ emitted, after optionally checking for the
285
+ presence of the type, when the parent
286
+ table is being created. When ``False``, no check
287
+ will be performed and no ``CREATE TYPE``
288
+ or ``DROP TYPE`` is emitted, unless
289
+ :meth:`~.postgresql.ENUM.create`
290
+ or :meth:`~.postgresql.ENUM.drop`
291
+ are called directly.
292
+ Setting to ``False`` is helpful
293
+ when invoking a creation scheme to a SQL file
294
+ without access to the actual database -
295
+ the :meth:`~.postgresql.ENUM.create` and
296
+ :meth:`~.postgresql.ENUM.drop` methods can
297
+ be used to emit SQL to a target bind.
298
+
299
+ """
300
+ native_enum = kw.pop("native_enum", None)
301
+ if native_enum is False:
302
+ util.warn(
303
+ "the native_enum flag does not apply to the "
304
+ "sqlalchemy.dialects.postgresql.ENUM datatype; this type "
305
+ "always refers to ENUM. Use sqlalchemy.types.Enum for "
306
+ "non-native enum."
307
+ )
308
+ if name is not _NoArg.NO_ARG:
309
+ kw["name"] = name
310
+ kw["create_type"] = create_type
311
+ super().__init__(*enums, **kw)
312
+
313
+ def coerce_compared_value(self, op, value):
314
+ super_coerced_type = super().coerce_compared_value(op, value)
315
+ if (
316
+ super_coerced_type._type_affinity
317
+ is type_api.STRINGTYPE._type_affinity
318
+ ):
319
+ return self
320
+ else:
321
+ return super_coerced_type
322
+
323
+ @classmethod
324
+ def __test_init__(cls):
325
+ return cls(name="name")
326
+
327
+ @classmethod
328
+ def adapt_emulated_to_native(cls, impl, **kw):
329
+ """Produce a PostgreSQL native :class:`_postgresql.ENUM` from plain
330
+ :class:`.Enum`.
331
+
332
+ """
333
+ kw.setdefault("validate_strings", impl.validate_strings)
334
+ kw.setdefault("name", impl.name)
335
+ kw.setdefault("create_type", impl.create_type)
336
+ kw.setdefault("schema", impl.schema)
337
+ kw.setdefault("metadata", impl.metadata)
338
+ kw.setdefault("_create_events", False)
339
+ kw.setdefault("values_callable", impl.values_callable)
340
+ kw.setdefault("omit_aliases", impl._omit_aliases)
341
+ kw.setdefault("_adapted_from", impl)
342
+
343
+ return cls(**kw)
344
+
345
+ def create(self, bind: _CreateDropBind, checkfirst: bool = True) -> None:
346
+ """Emit ``CREATE TYPE`` for this
347
+ :class:`_postgresql.ENUM`.
348
+
349
+ If the underlying dialect does not support
350
+ PostgreSQL CREATE TYPE, no action is taken.
351
+
352
+ :param bind: a connectable :class:`_engine.Engine`,
353
+ :class:`_engine.Connection`, or similar object to emit
354
+ SQL.
355
+ :param checkfirst: if ``True``, a query against
356
+ the PG catalog will be first performed to see
357
+ if the type does not exist already before
358
+ creating.
359
+
360
+ """
361
+ if not bind.dialect.supports_native_enum:
362
+ return
363
+
364
+ super().create(bind, checkfirst=checkfirst)
365
+
366
+ def drop(self, bind: _CreateDropBind, checkfirst: bool = True) -> None:
367
+ """Emit ``DROP TYPE`` for this
368
+ :class:`_postgresql.ENUM`.
369
+
370
+ If the underlying dialect does not support
371
+ PostgreSQL DROP TYPE, no action is taken.
372
+
373
+ :param bind: a connectable :class:`_engine.Engine`,
374
+ :class:`_engine.Connection`, or similar object to emit
375
+ SQL.
376
+ :param checkfirst: if ``True``, a query against
377
+ the PG catalog will be first performed to see
378
+ if the type actually exists before dropping.
379
+
380
+ """
381
+ if not bind.dialect.supports_native_enum:
382
+ return
383
+
384
+ super().drop(bind, checkfirst=checkfirst)
385
+
386
+ def get_dbapi_type(self, dbapi: ModuleType) -> None:
387
+ """dont return dbapi.STRING for ENUM in PostgreSQL, since that's
388
+ a different type"""
389
+
390
+ return None
391
+
392
+
393
+ class DomainGenerator(NamedTypeGenerator):
394
+ def visit_DOMAIN(self, domain):
395
+ if not self._can_create_type(domain):
396
+ return
397
+ with self.with_ddl_events(domain):
398
+ self.connection.execute(CreateDomainType(domain))
399
+
400
+
401
+ class DomainDropper(NamedTypeDropper):
402
+ def visit_DOMAIN(self, domain):
403
+ if not self._can_drop_type(domain):
404
+ return
405
+
406
+ with self.with_ddl_events(domain):
407
+ self.connection.execute(DropDomainType(domain))
408
+
409
+
410
+ class DOMAIN(NamedType, sqltypes.SchemaType):
411
+ r"""Represent the DOMAIN PostgreSQL type.
412
+
413
+ A domain is essentially a data type with optional constraints
414
+ that restrict the allowed set of values. E.g.::
415
+
416
+ PositiveInt = DOMAIN("pos_int", Integer, check="VALUE > 0", not_null=True)
417
+
418
+ UsPostalCode = DOMAIN(
419
+ "us_postal_code",
420
+ Text,
421
+ check="VALUE ~ '^\d{5}$' OR VALUE ~ '^\d{5}-\d{4}$'",
422
+ )
423
+
424
+ :class:`_postgresql.DOMAIN` has the same create/drop behavior specified
425
+ in :class:`_postgresql.ENUM`.
426
+
427
+ See the `PostgreSQL documentation`__ for additional details
428
+
429
+ __ https://www.postgresql.org/docs/current/sql-createdomain.html
430
+
431
+ .. versionadded:: 2.0
432
+
433
+ """ # noqa: E501
434
+
435
+ DDLGenerator = DomainGenerator
436
+ DDLDropper = DomainDropper
437
+
438
+ __visit_name__ = "DOMAIN"
439
+
440
+ def __init__(
441
+ self,
442
+ name: str,
443
+ data_type: _TypeEngineArgument[Any],
444
+ *,
445
+ collation: Optional[str] = None,
446
+ default: Union[elements.TextClause, str, None] = None,
447
+ constraint_name: Optional[str] = None,
448
+ not_null: Optional[bool] = None,
449
+ check: Union[elements.TextClause, str, None] = None,
450
+ create_type: bool = True,
451
+ **kw: Any,
452
+ ):
453
+ """
454
+ Construct a DOMAIN.
455
+
456
+ :param name: the name of the domain
457
+ :param data_type: The underlying data type of the domain.
458
+ This can include array specifiers.
459
+ :param collation: An optional collation for the domain.
460
+ If no collation is specified, the underlying data type's default
461
+ collation is used. The underlying type must be collatable if
462
+ ``collation`` is specified.
463
+ :param default: The DEFAULT clause specifies a default value for
464
+ columns of the domain data type. The default should be a string
465
+ or a :func:`_expression.text` value.
466
+ If no default value is specified, then the default value is
467
+ the null value.
468
+ :param constraint_name: An optional name for a constraint.
469
+ If not specified, the backend generates a name.
470
+ :param not_null: Values of this domain are prevented from being null.
471
+ By default domain are allowed to be null. If not specified
472
+ no nullability clause will be emitted.
473
+ :param check: CHECK clause specify integrity constraint or test
474
+ which values of the domain must satisfy. A constraint must be
475
+ an expression producing a Boolean result that can use the key
476
+ word VALUE to refer to the value being tested.
477
+ Differently from PostgreSQL, only a single check clause is
478
+ currently allowed in SQLAlchemy.
479
+ :param schema: optional schema name
480
+ :param metadata: optional :class:`_schema.MetaData` object which
481
+ this :class:`_postgresql.DOMAIN` will be directly associated
482
+ :param create_type: Defaults to True.
483
+ Indicates that ``CREATE TYPE`` should be emitted, after optionally
484
+ checking for the presence of the type, when the parent table is
485
+ being created; and additionally that ``DROP TYPE`` is called
486
+ when the table is dropped.
487
+
488
+ """
489
+ self.data_type = type_api.to_instance(data_type)
490
+ self.default = default
491
+ self.collation = collation
492
+ self.constraint_name = constraint_name
493
+ self.not_null = bool(not_null)
494
+ if check is not None:
495
+ check = coercions.expect(roles.DDLExpressionRole, check)
496
+ self.check = check
497
+ super().__init__(name=name, create_type=create_type, **kw)
498
+
499
+ @classmethod
500
+ def __test_init__(cls):
501
+ return cls("name", sqltypes.Integer)
502
+
503
+
504
+ class CreateEnumType(schema._CreateDropBase):
505
+ __visit_name__ = "create_enum_type"
506
+
507
+
508
+ class DropEnumType(schema._CreateDropBase):
509
+ __visit_name__ = "drop_enum_type"
510
+
511
+
512
+ class CreateDomainType(schema._CreateDropBase):
513
+ """Represent a CREATE DOMAIN statement."""
514
+
515
+ __visit_name__ = "create_domain_type"
516
+
517
+
518
+ class DropDomainType(schema._CreateDropBase):
519
+ """Represent a DROP DOMAIN statement."""
520
+
521
+ __visit_name__ = "drop_domain_type"
@@ -0,0 +1,130 @@
1
+ # dialects/postgresql/operators.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
+ # mypy: ignore-errors
8
+ from ...sql import operators
9
+
10
+
11
+ _getitem_precedence = operators._PRECEDENCE[operators.json_getitem_op]
12
+ _eq_precedence = operators._PRECEDENCE[operators.eq]
13
+
14
+ # JSON + JSONB
15
+ ASTEXT = operators.custom_op(
16
+ "->>",
17
+ precedence=_getitem_precedence,
18
+ natural_self_precedent=True,
19
+ eager_grouping=True,
20
+ )
21
+
22
+ JSONPATH_ASTEXT = operators.custom_op(
23
+ "#>>",
24
+ precedence=_getitem_precedence,
25
+ natural_self_precedent=True,
26
+ eager_grouping=True,
27
+ )
28
+
29
+ # JSONB + HSTORE
30
+ HAS_KEY = operators.custom_op(
31
+ "?",
32
+ precedence=_eq_precedence,
33
+ natural_self_precedent=True,
34
+ eager_grouping=True,
35
+ is_comparison=True,
36
+ )
37
+
38
+ HAS_ALL = operators.custom_op(
39
+ "?&",
40
+ precedence=_eq_precedence,
41
+ natural_self_precedent=True,
42
+ eager_grouping=True,
43
+ is_comparison=True,
44
+ )
45
+
46
+ HAS_ANY = operators.custom_op(
47
+ "?|",
48
+ precedence=_eq_precedence,
49
+ natural_self_precedent=True,
50
+ eager_grouping=True,
51
+ is_comparison=True,
52
+ )
53
+
54
+ # JSONB
55
+ DELETE_PATH = operators.custom_op(
56
+ "#-",
57
+ precedence=_getitem_precedence,
58
+ natural_self_precedent=True,
59
+ eager_grouping=True,
60
+ )
61
+
62
+ PATH_EXISTS = operators.custom_op(
63
+ "@?",
64
+ precedence=_eq_precedence,
65
+ natural_self_precedent=True,
66
+ eager_grouping=True,
67
+ is_comparison=True,
68
+ )
69
+
70
+ PATH_MATCH = operators.custom_op(
71
+ "@@",
72
+ precedence=_eq_precedence,
73
+ natural_self_precedent=True,
74
+ eager_grouping=True,
75
+ is_comparison=True,
76
+ )
77
+
78
+ # JSONB + ARRAY + HSTORE + RANGE
79
+ CONTAINS = operators.custom_op(
80
+ "@>",
81
+ precedence=_eq_precedence,
82
+ natural_self_precedent=True,
83
+ eager_grouping=True,
84
+ is_comparison=True,
85
+ )
86
+
87
+ CONTAINED_BY = operators.custom_op(
88
+ "<@",
89
+ precedence=_eq_precedence,
90
+ natural_self_precedent=True,
91
+ eager_grouping=True,
92
+ is_comparison=True,
93
+ )
94
+
95
+ # ARRAY + RANGE
96
+ OVERLAP = operators.custom_op(
97
+ "&&",
98
+ precedence=_eq_precedence,
99
+ is_comparison=True,
100
+ )
101
+
102
+ # RANGE
103
+ STRICTLY_LEFT_OF = operators.custom_op(
104
+ "<<", precedence=_eq_precedence, is_comparison=True
105
+ )
106
+
107
+ STRICTLY_RIGHT_OF = operators.custom_op(
108
+ ">>", precedence=_eq_precedence, is_comparison=True
109
+ )
110
+
111
+ NOT_EXTEND_RIGHT_OF = operators.custom_op(
112
+ "&<", precedence=_eq_precedence, is_comparison=True
113
+ )
114
+
115
+ NOT_EXTEND_LEFT_OF = operators.custom_op(
116
+ "&>", precedence=_eq_precedence, is_comparison=True
117
+ )
118
+
119
+ ADJACENT_TO = operators.custom_op(
120
+ "-|-", precedence=_eq_precedence, is_comparison=True
121
+ )
122
+
123
+ # HSTORE
124
+ GETITEM = operators.custom_op(
125
+ "->",
126
+ precedence=_getitem_precedence,
127
+ natural_self_precedent=True,
128
+ eager_grouping=True,
129
+ visit_name="hstore_getitem",
130
+ )