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,560 @@
1
+ # ext/declarative/extensions.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
+
9
+
10
+ """Public API functions and helpers for declarative."""
11
+ from __future__ import annotations
12
+
13
+ import collections
14
+ import contextlib
15
+ from typing import Any
16
+ from typing import Callable
17
+ from typing import TYPE_CHECKING
18
+ from typing import Union
19
+
20
+ from ... import exc as sa_exc
21
+ from ...engine import Connection
22
+ from ...engine import Engine
23
+ from ...orm import exc as orm_exc
24
+ from ...orm import relationships
25
+ from ...orm.base import _mapper_or_none
26
+ from ...orm.clsregistry import _resolver
27
+ from ...orm.decl_base import _DeferredDeclarativeConfig
28
+ from ...orm.util import polymorphic_union
29
+ from ...schema import Table
30
+ from ...util import OrderedDict
31
+
32
+ if TYPE_CHECKING:
33
+ from ...sql.schema import MetaData
34
+
35
+
36
+ class ConcreteBase:
37
+ """A helper class for 'concrete' declarative mappings.
38
+
39
+ :class:`.ConcreteBase` will use the :func:`.polymorphic_union`
40
+ function automatically, against all tables mapped as a subclass
41
+ to this class. The function is called via the
42
+ ``__declare_last__()`` function, which is essentially
43
+ a hook for the :meth:`.MapperEvents.after_configured` event.
44
+
45
+ :class:`.ConcreteBase` produces a mapped
46
+ table for the class itself. Compare to :class:`.AbstractConcreteBase`,
47
+ which does not.
48
+
49
+ Example::
50
+
51
+ from sqlalchemy.ext.declarative import ConcreteBase
52
+
53
+
54
+ class Employee(ConcreteBase, Base):
55
+ __tablename__ = "employee"
56
+ employee_id = Column(Integer, primary_key=True)
57
+ name = Column(String(50))
58
+ __mapper_args__ = {
59
+ "polymorphic_identity": "employee",
60
+ "concrete": True,
61
+ }
62
+
63
+
64
+ class Manager(Employee):
65
+ __tablename__ = "manager"
66
+ employee_id = Column(Integer, primary_key=True)
67
+ name = Column(String(50))
68
+ manager_data = Column(String(40))
69
+ __mapper_args__ = {
70
+ "polymorphic_identity": "manager",
71
+ "concrete": True,
72
+ }
73
+
74
+ The name of the discriminator column used by :func:`.polymorphic_union`
75
+ defaults to the name ``type``. To suit the use case of a mapping where an
76
+ actual column in a mapped table is already named ``type``, the
77
+ discriminator name can be configured by setting the
78
+ ``_concrete_discriminator_name`` attribute::
79
+
80
+ class Employee(ConcreteBase, Base):
81
+ _concrete_discriminator_name = "_concrete_discriminator"
82
+
83
+ .. versionchanged:: 1.4.2 The ``_concrete_discriminator_name`` attribute
84
+ need only be placed on the basemost class to take correct effect for
85
+ all subclasses. An explicit error message is now raised if the
86
+ mapped column names conflict with the discriminator name, whereas
87
+ in the 1.3.x series there would be some warnings and then a non-useful
88
+ query would be generated.
89
+
90
+ .. seealso::
91
+
92
+ :class:`.AbstractConcreteBase`
93
+
94
+ :ref:`concrete_inheritance`
95
+
96
+
97
+ """
98
+
99
+ @classmethod
100
+ def _create_polymorphic_union(cls, mappers, discriminator_name):
101
+ return polymorphic_union(
102
+ OrderedDict(
103
+ (mp.polymorphic_identity, mp.local_table) for mp in mappers
104
+ ),
105
+ discriminator_name,
106
+ "pjoin",
107
+ )
108
+
109
+ @classmethod
110
+ def __declare_first__(cls):
111
+ m = cls.__mapper__
112
+ if m.with_polymorphic:
113
+ return
114
+
115
+ discriminator_name = (
116
+ getattr(cls, "_concrete_discriminator_name", None) or "type"
117
+ )
118
+
119
+ mappers = list(m.self_and_descendants)
120
+ pjoin = cls._create_polymorphic_union(mappers, discriminator_name)
121
+ m._set_with_polymorphic(("*", pjoin))
122
+ m._set_polymorphic_on(pjoin.c[discriminator_name])
123
+
124
+
125
+ class AbstractConcreteBase(ConcreteBase):
126
+ """A helper class for 'concrete' declarative mappings.
127
+
128
+ :class:`.AbstractConcreteBase` will use the :func:`.polymorphic_union`
129
+ function automatically, against all tables mapped as a subclass
130
+ to this class. The function is called via the
131
+ ``__declare_first__()`` function, which is essentially
132
+ a hook for the :meth:`.MapperEvents.before_configured` event.
133
+
134
+ :class:`.AbstractConcreteBase` applies :class:`_orm.Mapper` for its
135
+ immediately inheriting class, as would occur for any other
136
+ declarative mapped class. However, the :class:`_orm.Mapper` is not
137
+ mapped to any particular :class:`.Table` object. Instead, it's
138
+ mapped directly to the "polymorphic" selectable produced by
139
+ :func:`.polymorphic_union`, and performs no persistence operations on its
140
+ own. Compare to :class:`.ConcreteBase`, which maps its
141
+ immediately inheriting class to an actual
142
+ :class:`.Table` that stores rows directly.
143
+
144
+ .. note::
145
+
146
+ The :class:`.AbstractConcreteBase` delays the mapper creation of the
147
+ base class until all the subclasses have been defined,
148
+ as it needs to create a mapping against a selectable that will include
149
+ all subclass tables. In order to achieve this, it waits for the
150
+ **mapper configuration event** to occur, at which point it scans
151
+ through all the configured subclasses and sets up a mapping that will
152
+ query against all subclasses at once.
153
+
154
+ While this event is normally invoked automatically, in the case of
155
+ :class:`.AbstractConcreteBase`, it may be necessary to invoke it
156
+ explicitly after **all** subclass mappings are defined, if the first
157
+ operation is to be a query against this base class. To do so, once all
158
+ the desired classes have been configured, the
159
+ :meth:`_orm.registry.configure` method on the :class:`_orm.registry`
160
+ in use can be invoked, which is available in relation to a particular
161
+ declarative base class::
162
+
163
+ Base.registry.configure()
164
+
165
+ Example::
166
+
167
+ from sqlalchemy.orm import DeclarativeBase
168
+ from sqlalchemy.ext.declarative import AbstractConcreteBase
169
+
170
+
171
+ class Base(DeclarativeBase):
172
+ pass
173
+
174
+
175
+ class Employee(AbstractConcreteBase, Base):
176
+ pass
177
+
178
+
179
+ class Manager(Employee):
180
+ __tablename__ = "manager"
181
+ employee_id = Column(Integer, primary_key=True)
182
+ name = Column(String(50))
183
+ manager_data = Column(String(40))
184
+
185
+ __mapper_args__ = {
186
+ "polymorphic_identity": "manager",
187
+ "concrete": True,
188
+ }
189
+
190
+
191
+ Base.registry.configure()
192
+
193
+ The abstract base class is handled by declarative in a special way;
194
+ at class configuration time, it behaves like a declarative mixin
195
+ or an ``__abstract__`` base class. Once classes are configured
196
+ and mappings are produced, it then gets mapped itself, but
197
+ after all of its descendants. This is a very unique system of mapping
198
+ not found in any other SQLAlchemy API feature.
199
+
200
+ Using this approach, we can specify columns and properties
201
+ that will take place on mapped subclasses, in the way that
202
+ we normally do as in :ref:`declarative_mixins`::
203
+
204
+ from sqlalchemy.ext.declarative import AbstractConcreteBase
205
+
206
+
207
+ class Company(Base):
208
+ __tablename__ = "company"
209
+ id = Column(Integer, primary_key=True)
210
+
211
+
212
+ class Employee(AbstractConcreteBase, Base):
213
+ strict_attrs = True
214
+
215
+ employee_id = Column(Integer, primary_key=True)
216
+
217
+ @declared_attr
218
+ def company_id(cls):
219
+ return Column(ForeignKey("company.id"))
220
+
221
+ @declared_attr
222
+ def company(cls):
223
+ return relationship("Company")
224
+
225
+
226
+ class Manager(Employee):
227
+ __tablename__ = "manager"
228
+
229
+ name = Column(String(50))
230
+ manager_data = Column(String(40))
231
+
232
+ __mapper_args__ = {
233
+ "polymorphic_identity": "manager",
234
+ "concrete": True,
235
+ }
236
+
237
+
238
+ Base.registry.configure()
239
+
240
+ When we make use of our mappings however, both ``Manager`` and
241
+ ``Employee`` will have an independently usable ``.company`` attribute::
242
+
243
+ session.execute(select(Employee).filter(Employee.company.has(id=5)))
244
+
245
+ :param strict_attrs: when specified on the base class, "strict" attribute
246
+ mode is enabled which attempts to limit ORM mapped attributes on the
247
+ base class to only those that are immediately present, while still
248
+ preserving "polymorphic" loading behavior.
249
+
250
+ .. versionadded:: 2.0
251
+
252
+ .. seealso::
253
+
254
+ :class:`.ConcreteBase`
255
+
256
+ :ref:`concrete_inheritance`
257
+
258
+ :ref:`abstract_concrete_base`
259
+
260
+ """
261
+
262
+ __no_table__ = True
263
+
264
+ @classmethod
265
+ def __declare_first__(cls):
266
+ cls._sa_decl_prepare_nocascade()
267
+
268
+ @classmethod
269
+ def _sa_decl_prepare_nocascade(cls):
270
+ if getattr(cls, "__mapper__", None):
271
+ return
272
+
273
+ to_map = _DeferredDeclarativeConfig.config_for_cls(cls)
274
+
275
+ # can't rely on 'self_and_descendants' here
276
+ # since technically an immediate subclass
277
+ # might not be mapped, but a subclass
278
+ # may be.
279
+ mappers = []
280
+ stack = list(cls.__subclasses__())
281
+ while stack:
282
+ klass = stack.pop()
283
+ stack.extend(klass.__subclasses__())
284
+ mn = _mapper_or_none(klass)
285
+ if mn is not None:
286
+ mappers.append(mn)
287
+
288
+ discriminator_name = (
289
+ getattr(cls, "_concrete_discriminator_name", None) or "type"
290
+ )
291
+ pjoin = cls._create_polymorphic_union(mappers, discriminator_name)
292
+
293
+ # For columns that were declared on the class, these
294
+ # are normally ignored with the "__no_table__" mapping,
295
+ # unless they have a different attribute key vs. col name
296
+ # and are in the properties argument.
297
+ # In that case, ensure we update the properties entry
298
+ # to the correct column from the pjoin target table.
299
+ declared_cols = set(to_map.declared_columns)
300
+ declared_col_keys = {c.key for c in declared_cols}
301
+ for k, v in list(to_map.properties.items()):
302
+ if v in declared_cols:
303
+ to_map.properties[k] = pjoin.c[v.key]
304
+ declared_col_keys.remove(v.key)
305
+
306
+ to_map.local_table = pjoin
307
+
308
+ strict_attrs = cls.__dict__.get("strict_attrs", False)
309
+
310
+ m_args = to_map.mapper_args_fn or dict
311
+
312
+ def mapper_args():
313
+ args = m_args()
314
+ args["polymorphic_on"] = pjoin.c[discriminator_name]
315
+ args["polymorphic_abstract"] = True
316
+ if strict_attrs:
317
+ args["include_properties"] = (
318
+ set(pjoin.primary_key)
319
+ | declared_col_keys
320
+ | {discriminator_name}
321
+ )
322
+ args["with_polymorphic"] = ("*", pjoin)
323
+ return args
324
+
325
+ to_map.mapper_args_fn = mapper_args
326
+
327
+ to_map.map()
328
+
329
+ stack = [cls]
330
+ while stack:
331
+ scls = stack.pop(0)
332
+ stack.extend(scls.__subclasses__())
333
+ sm = _mapper_or_none(scls)
334
+ if sm and sm.concrete and sm.inherits is None:
335
+ for sup_ in scls.__mro__[1:]:
336
+ sup_sm = _mapper_or_none(sup_)
337
+ if sup_sm:
338
+ sm._set_concrete_base(sup_sm)
339
+ break
340
+
341
+ @classmethod
342
+ def _sa_raise_deferred_config(cls):
343
+ raise orm_exc.UnmappedClassError(
344
+ cls,
345
+ msg="Class %s is a subclass of AbstractConcreteBase and "
346
+ "has a mapping pending until all subclasses are defined. "
347
+ "Call the sqlalchemy.orm.configure_mappers() function after "
348
+ "all subclasses have been defined to "
349
+ "complete the mapping of this class."
350
+ % orm_exc._safe_cls_name(cls),
351
+ )
352
+
353
+
354
+ class DeferredReflection:
355
+ """A helper class for construction of mappings based on
356
+ a deferred reflection step.
357
+
358
+ Normally, declarative can be used with reflection by
359
+ setting a :class:`_schema.Table` object using autoload_with=engine
360
+ as the ``__table__`` attribute on a declarative class.
361
+ The caveat is that the :class:`_schema.Table` must be fully
362
+ reflected, or at the very least have a primary key column,
363
+ at the point at which a normal declarative mapping is
364
+ constructed, meaning the :class:`_engine.Engine` must be available
365
+ at class declaration time.
366
+
367
+ The :class:`.DeferredReflection` mixin moves the construction
368
+ of mappers to be at a later point, after a specific
369
+ method is called which first reflects all :class:`_schema.Table`
370
+ objects created so far. Classes can define it as such::
371
+
372
+ from sqlalchemy.ext.declarative import declarative_base
373
+ from sqlalchemy.ext.declarative import DeferredReflection
374
+
375
+ Base = declarative_base()
376
+
377
+
378
+ class MyClass(DeferredReflection, Base):
379
+ __tablename__ = "mytable"
380
+
381
+ Above, ``MyClass`` is not yet mapped. After a series of
382
+ classes have been defined in the above fashion, all tables
383
+ can be reflected and mappings created using
384
+ :meth:`.prepare`::
385
+
386
+ engine = create_engine("someengine://...")
387
+ DeferredReflection.prepare(engine)
388
+
389
+ The :class:`.DeferredReflection` mixin can be applied to individual
390
+ classes, used as the base for the declarative base itself,
391
+ or used in a custom abstract class. Using an abstract base
392
+ allows that only a subset of classes to be prepared for a
393
+ particular prepare step, which is necessary for applications
394
+ that use more than one engine. For example, if an application
395
+ has two engines, you might use two bases, and prepare each
396
+ separately, e.g.::
397
+
398
+ class ReflectedOne(DeferredReflection, Base):
399
+ __abstract__ = True
400
+
401
+
402
+ class ReflectedTwo(DeferredReflection, Base):
403
+ __abstract__ = True
404
+
405
+
406
+ class MyClass(ReflectedOne):
407
+ __tablename__ = "mytable"
408
+
409
+
410
+ class MyOtherClass(ReflectedOne):
411
+ __tablename__ = "myothertable"
412
+
413
+
414
+ class YetAnotherClass(ReflectedTwo):
415
+ __tablename__ = "yetanothertable"
416
+
417
+
418
+ # ... etc.
419
+
420
+ Above, the class hierarchies for ``ReflectedOne`` and
421
+ ``ReflectedTwo`` can be configured separately::
422
+
423
+ ReflectedOne.prepare(engine_one)
424
+ ReflectedTwo.prepare(engine_two)
425
+
426
+ .. seealso::
427
+
428
+ :ref:`orm_declarative_reflected_deferred_reflection` - in the
429
+ :ref:`orm_declarative_table_config_toplevel` section.
430
+
431
+ """
432
+
433
+ @classmethod
434
+ def prepare(
435
+ cls, bind: Union[Engine, Connection], **reflect_kw: Any
436
+ ) -> None:
437
+ r"""Reflect all :class:`_schema.Table` objects for all current
438
+ :class:`.DeferredReflection` subclasses
439
+
440
+ :param bind: :class:`_engine.Engine` or :class:`_engine.Connection`
441
+ instance
442
+
443
+ ..versionchanged:: 2.0.16 a :class:`_engine.Connection` is also
444
+ accepted.
445
+
446
+ :param \**reflect_kw: additional keyword arguments passed to
447
+ :meth:`_schema.MetaData.reflect`, such as
448
+ :paramref:`_schema.MetaData.reflect.views`.
449
+
450
+ .. versionadded:: 2.0.16
451
+
452
+ """
453
+
454
+ to_map = _DeferredDeclarativeConfig.classes_for_base(cls)
455
+
456
+ metadata_to_table = collections.defaultdict(set)
457
+
458
+ # first collect the primary __table__ for each class into a
459
+ # collection of metadata/schemaname -> table names
460
+ for thingy in to_map:
461
+ if thingy.local_table is not None:
462
+ metadata_to_table[
463
+ (thingy.local_table.metadata, thingy.local_table.schema)
464
+ ].add(thingy.local_table.name)
465
+
466
+ # then reflect all those tables into their metadatas
467
+
468
+ if isinstance(bind, Connection):
469
+ conn = bind
470
+ ctx = contextlib.nullcontext(enter_result=conn)
471
+ elif isinstance(bind, Engine):
472
+ ctx = bind.connect()
473
+ else:
474
+ raise sa_exc.ArgumentError(
475
+ f"Expected Engine or Connection, got {bind!r}"
476
+ )
477
+
478
+ with ctx as conn:
479
+ for (metadata, schema), table_names in metadata_to_table.items():
480
+ metadata.reflect(
481
+ conn,
482
+ only=table_names,
483
+ schema=schema,
484
+ extend_existing=True,
485
+ autoload_replace=False,
486
+ **reflect_kw,
487
+ )
488
+
489
+ metadata_to_table.clear()
490
+
491
+ # .map() each class, then go through relationships and look
492
+ # for secondary
493
+ for thingy in to_map:
494
+ thingy.map()
495
+
496
+ mapper = thingy.cls.__mapper__
497
+ metadata = mapper.class_.metadata
498
+
499
+ for rel in mapper._props.values():
500
+ if (
501
+ isinstance(rel, relationships.RelationshipProperty)
502
+ and rel._init_args.secondary._is_populated()
503
+ ):
504
+ secondary_arg = rel._init_args.secondary
505
+
506
+ if isinstance(secondary_arg.argument, Table):
507
+ secondary_table = secondary_arg.argument
508
+ metadata_to_table[
509
+ (
510
+ secondary_table.metadata,
511
+ secondary_table.schema,
512
+ )
513
+ ].add(secondary_table.name)
514
+ elif isinstance(secondary_arg.argument, str):
515
+ _, resolve_arg = _resolver(rel.parent.class_, rel)
516
+
517
+ resolver = resolve_arg(
518
+ secondary_arg.argument, True
519
+ )
520
+ metadata_to_table[
521
+ (metadata, thingy.local_table.schema)
522
+ ].add(secondary_arg.argument)
523
+
524
+ resolver._resolvers += (
525
+ cls._sa_deferred_table_resolver(metadata),
526
+ )
527
+
528
+ secondary_arg.argument = resolver()
529
+
530
+ for (metadata, schema), table_names in metadata_to_table.items():
531
+ metadata.reflect(
532
+ conn,
533
+ only=table_names,
534
+ schema=schema,
535
+ extend_existing=True,
536
+ autoload_replace=False,
537
+ )
538
+
539
+ @classmethod
540
+ def _sa_deferred_table_resolver(
541
+ cls, metadata: MetaData
542
+ ) -> Callable[[str], Table]:
543
+ def _resolve(key: str) -> Table:
544
+ # reflection has already occurred so this Table would have
545
+ # its contents already
546
+ return Table(key, metadata)
547
+
548
+ return _resolve
549
+
550
+ _sa_decl_prepare = True
551
+
552
+ @classmethod
553
+ def _sa_raise_deferred_config(cls):
554
+ raise orm_exc.UnmappedClassError(
555
+ cls,
556
+ msg="Class %s is a subclass of DeferredReflection. "
557
+ "Mappings are not produced until the .prepare() "
558
+ "method is called on the class hierarchy."
559
+ % orm_exc._safe_cls_name(cls),
560
+ )