SQLAlchemy 2.0.47__cp313-cp313t-win32.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 (274) hide show
  1. sqlalchemy/__init__.py +283 -0
  2. sqlalchemy/connectors/__init__.py +18 -0
  3. sqlalchemy/connectors/aioodbc.py +184 -0
  4. sqlalchemy/connectors/asyncio.py +429 -0
  5. sqlalchemy/connectors/pyodbc.py +250 -0
  6. sqlalchemy/cyextension/__init__.py +6 -0
  7. sqlalchemy/cyextension/collections.cp313t-win32.pyd +0 -0
  8. sqlalchemy/cyextension/collections.pyx +409 -0
  9. sqlalchemy/cyextension/immutabledict.cp313t-win32.pyd +0 -0
  10. sqlalchemy/cyextension/immutabledict.pxd +8 -0
  11. sqlalchemy/cyextension/immutabledict.pyx +133 -0
  12. sqlalchemy/cyextension/processors.cp313t-win32.pyd +0 -0
  13. sqlalchemy/cyextension/processors.pyx +68 -0
  14. sqlalchemy/cyextension/resultproxy.cp313t-win32.pyd +0 -0
  15. sqlalchemy/cyextension/resultproxy.pyx +102 -0
  16. sqlalchemy/cyextension/util.cp313t-win32.pyd +0 -0
  17. sqlalchemy/cyextension/util.pyx +90 -0
  18. sqlalchemy/dialects/__init__.py +62 -0
  19. sqlalchemy/dialects/_typing.py +30 -0
  20. sqlalchemy/dialects/mssql/__init__.py +88 -0
  21. sqlalchemy/dialects/mssql/aioodbc.py +63 -0
  22. sqlalchemy/dialects/mssql/base.py +4093 -0
  23. sqlalchemy/dialects/mssql/information_schema.py +285 -0
  24. sqlalchemy/dialects/mssql/json.py +129 -0
  25. sqlalchemy/dialects/mssql/provision.py +185 -0
  26. sqlalchemy/dialects/mssql/pymssql.py +126 -0
  27. sqlalchemy/dialects/mssql/pyodbc.py +760 -0
  28. sqlalchemy/dialects/mysql/__init__.py +104 -0
  29. sqlalchemy/dialects/mysql/aiomysql.py +250 -0
  30. sqlalchemy/dialects/mysql/asyncmy.py +231 -0
  31. sqlalchemy/dialects/mysql/base.py +3949 -0
  32. sqlalchemy/dialects/mysql/cymysql.py +106 -0
  33. sqlalchemy/dialects/mysql/dml.py +225 -0
  34. sqlalchemy/dialects/mysql/enumerated.py +282 -0
  35. sqlalchemy/dialects/mysql/expression.py +146 -0
  36. sqlalchemy/dialects/mysql/json.py +91 -0
  37. sqlalchemy/dialects/mysql/mariadb.py +72 -0
  38. sqlalchemy/dialects/mysql/mariadbconnector.py +322 -0
  39. sqlalchemy/dialects/mysql/mysqlconnector.py +302 -0
  40. sqlalchemy/dialects/mysql/mysqldb.py +314 -0
  41. sqlalchemy/dialects/mysql/provision.py +153 -0
  42. sqlalchemy/dialects/mysql/pymysql.py +158 -0
  43. sqlalchemy/dialects/mysql/pyodbc.py +157 -0
  44. sqlalchemy/dialects/mysql/reflection.py +727 -0
  45. sqlalchemy/dialects/mysql/reserved_words.py +570 -0
  46. sqlalchemy/dialects/mysql/types.py +835 -0
  47. sqlalchemy/dialects/oracle/__init__.py +81 -0
  48. sqlalchemy/dialects/oracle/base.py +3802 -0
  49. sqlalchemy/dialects/oracle/cx_oracle.py +1555 -0
  50. sqlalchemy/dialects/oracle/dictionary.py +507 -0
  51. sqlalchemy/dialects/oracle/oracledb.py +941 -0
  52. sqlalchemy/dialects/oracle/provision.py +297 -0
  53. sqlalchemy/dialects/oracle/types.py +316 -0
  54. sqlalchemy/dialects/oracle/vector.py +365 -0
  55. sqlalchemy/dialects/postgresql/__init__.py +167 -0
  56. sqlalchemy/dialects/postgresql/_psycopg_common.py +189 -0
  57. sqlalchemy/dialects/postgresql/array.py +519 -0
  58. sqlalchemy/dialects/postgresql/asyncpg.py +1284 -0
  59. sqlalchemy/dialects/postgresql/base.py +5378 -0
  60. sqlalchemy/dialects/postgresql/dml.py +339 -0
  61. sqlalchemy/dialects/postgresql/ext.py +540 -0
  62. sqlalchemy/dialects/postgresql/hstore.py +406 -0
  63. sqlalchemy/dialects/postgresql/json.py +404 -0
  64. sqlalchemy/dialects/postgresql/named_types.py +524 -0
  65. sqlalchemy/dialects/postgresql/operators.py +129 -0
  66. sqlalchemy/dialects/postgresql/pg8000.py +669 -0
  67. sqlalchemy/dialects/postgresql/pg_catalog.py +326 -0
  68. sqlalchemy/dialects/postgresql/provision.py +183 -0
  69. sqlalchemy/dialects/postgresql/psycopg.py +862 -0
  70. sqlalchemy/dialects/postgresql/psycopg2.py +892 -0
  71. sqlalchemy/dialects/postgresql/psycopg2cffi.py +61 -0
  72. sqlalchemy/dialects/postgresql/ranges.py +1031 -0
  73. sqlalchemy/dialects/postgresql/types.py +313 -0
  74. sqlalchemy/dialects/sqlite/__init__.py +57 -0
  75. sqlalchemy/dialects/sqlite/aiosqlite.py +482 -0
  76. sqlalchemy/dialects/sqlite/base.py +3056 -0
  77. sqlalchemy/dialects/sqlite/dml.py +263 -0
  78. sqlalchemy/dialects/sqlite/json.py +92 -0
  79. sqlalchemy/dialects/sqlite/provision.py +229 -0
  80. sqlalchemy/dialects/sqlite/pysqlcipher.py +157 -0
  81. sqlalchemy/dialects/sqlite/pysqlite.py +756 -0
  82. sqlalchemy/dialects/type_migration_guidelines.txt +145 -0
  83. sqlalchemy/engine/__init__.py +62 -0
  84. sqlalchemy/engine/_py_processors.py +136 -0
  85. sqlalchemy/engine/_py_row.py +128 -0
  86. sqlalchemy/engine/_py_util.py +74 -0
  87. sqlalchemy/engine/base.py +3390 -0
  88. sqlalchemy/engine/characteristics.py +155 -0
  89. sqlalchemy/engine/create.py +893 -0
  90. sqlalchemy/engine/cursor.py +2298 -0
  91. sqlalchemy/engine/default.py +2394 -0
  92. sqlalchemy/engine/events.py +965 -0
  93. sqlalchemy/engine/interfaces.py +3471 -0
  94. sqlalchemy/engine/mock.py +134 -0
  95. sqlalchemy/engine/processors.py +61 -0
  96. sqlalchemy/engine/reflection.py +2102 -0
  97. sqlalchemy/engine/result.py +2399 -0
  98. sqlalchemy/engine/row.py +400 -0
  99. sqlalchemy/engine/strategies.py +16 -0
  100. sqlalchemy/engine/url.py +924 -0
  101. sqlalchemy/engine/util.py +167 -0
  102. sqlalchemy/event/__init__.py +26 -0
  103. sqlalchemy/event/api.py +220 -0
  104. sqlalchemy/event/attr.py +676 -0
  105. sqlalchemy/event/base.py +472 -0
  106. sqlalchemy/event/legacy.py +258 -0
  107. sqlalchemy/event/registry.py +390 -0
  108. sqlalchemy/events.py +17 -0
  109. sqlalchemy/exc.py +832 -0
  110. sqlalchemy/ext/__init__.py +11 -0
  111. sqlalchemy/ext/associationproxy.py +2027 -0
  112. sqlalchemy/ext/asyncio/__init__.py +25 -0
  113. sqlalchemy/ext/asyncio/base.py +281 -0
  114. sqlalchemy/ext/asyncio/engine.py +1471 -0
  115. sqlalchemy/ext/asyncio/exc.py +21 -0
  116. sqlalchemy/ext/asyncio/result.py +965 -0
  117. sqlalchemy/ext/asyncio/scoping.py +1599 -0
  118. sqlalchemy/ext/asyncio/session.py +1947 -0
  119. sqlalchemy/ext/automap.py +1701 -0
  120. sqlalchemy/ext/baked.py +570 -0
  121. sqlalchemy/ext/compiler.py +600 -0
  122. sqlalchemy/ext/declarative/__init__.py +65 -0
  123. sqlalchemy/ext/declarative/extensions.py +564 -0
  124. sqlalchemy/ext/horizontal_shard.py +478 -0
  125. sqlalchemy/ext/hybrid.py +1535 -0
  126. sqlalchemy/ext/indexable.py +364 -0
  127. sqlalchemy/ext/instrumentation.py +450 -0
  128. sqlalchemy/ext/mutable.py +1085 -0
  129. sqlalchemy/ext/mypy/__init__.py +6 -0
  130. sqlalchemy/ext/mypy/apply.py +324 -0
  131. sqlalchemy/ext/mypy/decl_class.py +515 -0
  132. sqlalchemy/ext/mypy/infer.py +590 -0
  133. sqlalchemy/ext/mypy/names.py +335 -0
  134. sqlalchemy/ext/mypy/plugin.py +303 -0
  135. sqlalchemy/ext/mypy/util.py +357 -0
  136. sqlalchemy/ext/orderinglist.py +439 -0
  137. sqlalchemy/ext/serializer.py +185 -0
  138. sqlalchemy/future/__init__.py +16 -0
  139. sqlalchemy/future/engine.py +15 -0
  140. sqlalchemy/inspection.py +174 -0
  141. sqlalchemy/log.py +288 -0
  142. sqlalchemy/orm/__init__.py +171 -0
  143. sqlalchemy/orm/_orm_constructors.py +2661 -0
  144. sqlalchemy/orm/_typing.py +179 -0
  145. sqlalchemy/orm/attributes.py +2845 -0
  146. sqlalchemy/orm/base.py +971 -0
  147. sqlalchemy/orm/bulk_persistence.py +2135 -0
  148. sqlalchemy/orm/clsregistry.py +571 -0
  149. sqlalchemy/orm/collections.py +1627 -0
  150. sqlalchemy/orm/context.py +3334 -0
  151. sqlalchemy/orm/decl_api.py +2004 -0
  152. sqlalchemy/orm/decl_base.py +2192 -0
  153. sqlalchemy/orm/dependency.py +1302 -0
  154. sqlalchemy/orm/descriptor_props.py +1092 -0
  155. sqlalchemy/orm/dynamic.py +300 -0
  156. sqlalchemy/orm/evaluator.py +379 -0
  157. sqlalchemy/orm/events.py +3252 -0
  158. sqlalchemy/orm/exc.py +237 -0
  159. sqlalchemy/orm/identity.py +302 -0
  160. sqlalchemy/orm/instrumentation.py +754 -0
  161. sqlalchemy/orm/interfaces.py +1496 -0
  162. sqlalchemy/orm/loading.py +1686 -0
  163. sqlalchemy/orm/mapped_collection.py +557 -0
  164. sqlalchemy/orm/mapper.py +4444 -0
  165. sqlalchemy/orm/path_registry.py +809 -0
  166. sqlalchemy/orm/persistence.py +1788 -0
  167. sqlalchemy/orm/properties.py +935 -0
  168. sqlalchemy/orm/query.py +3459 -0
  169. sqlalchemy/orm/relationships.py +3508 -0
  170. sqlalchemy/orm/scoping.py +2148 -0
  171. sqlalchemy/orm/session.py +5280 -0
  172. sqlalchemy/orm/state.py +1168 -0
  173. sqlalchemy/orm/state_changes.py +196 -0
  174. sqlalchemy/orm/strategies.py +3470 -0
  175. sqlalchemy/orm/strategy_options.py +2568 -0
  176. sqlalchemy/orm/sync.py +164 -0
  177. sqlalchemy/orm/unitofwork.py +796 -0
  178. sqlalchemy/orm/util.py +2403 -0
  179. sqlalchemy/orm/writeonly.py +674 -0
  180. sqlalchemy/pool/__init__.py +44 -0
  181. sqlalchemy/pool/base.py +1524 -0
  182. sqlalchemy/pool/events.py +375 -0
  183. sqlalchemy/pool/impl.py +588 -0
  184. sqlalchemy/py.typed +0 -0
  185. sqlalchemy/schema.py +69 -0
  186. sqlalchemy/sql/__init__.py +145 -0
  187. sqlalchemy/sql/_dml_constructors.py +132 -0
  188. sqlalchemy/sql/_elements_constructors.py +1872 -0
  189. sqlalchemy/sql/_orm_types.py +20 -0
  190. sqlalchemy/sql/_py_util.py +75 -0
  191. sqlalchemy/sql/_selectable_constructors.py +763 -0
  192. sqlalchemy/sql/_typing.py +482 -0
  193. sqlalchemy/sql/annotation.py +587 -0
  194. sqlalchemy/sql/base.py +2293 -0
  195. sqlalchemy/sql/cache_key.py +1057 -0
  196. sqlalchemy/sql/coercions.py +1404 -0
  197. sqlalchemy/sql/compiler.py +8081 -0
  198. sqlalchemy/sql/crud.py +1752 -0
  199. sqlalchemy/sql/ddl.py +1444 -0
  200. sqlalchemy/sql/default_comparator.py +551 -0
  201. sqlalchemy/sql/dml.py +1850 -0
  202. sqlalchemy/sql/elements.py +5589 -0
  203. sqlalchemy/sql/events.py +458 -0
  204. sqlalchemy/sql/expression.py +159 -0
  205. sqlalchemy/sql/functions.py +2158 -0
  206. sqlalchemy/sql/lambdas.py +1442 -0
  207. sqlalchemy/sql/naming.py +209 -0
  208. sqlalchemy/sql/operators.py +2623 -0
  209. sqlalchemy/sql/roles.py +323 -0
  210. sqlalchemy/sql/schema.py +6222 -0
  211. sqlalchemy/sql/selectable.py +7265 -0
  212. sqlalchemy/sql/sqltypes.py +3930 -0
  213. sqlalchemy/sql/traversals.py +1024 -0
  214. sqlalchemy/sql/type_api.py +2368 -0
  215. sqlalchemy/sql/util.py +1485 -0
  216. sqlalchemy/sql/visitors.py +1164 -0
  217. sqlalchemy/testing/__init__.py +96 -0
  218. sqlalchemy/testing/assertions.py +994 -0
  219. sqlalchemy/testing/assertsql.py +520 -0
  220. sqlalchemy/testing/asyncio.py +135 -0
  221. sqlalchemy/testing/config.py +434 -0
  222. sqlalchemy/testing/engines.py +483 -0
  223. sqlalchemy/testing/entities.py +117 -0
  224. sqlalchemy/testing/exclusions.py +476 -0
  225. sqlalchemy/testing/fixtures/__init__.py +28 -0
  226. sqlalchemy/testing/fixtures/base.py +384 -0
  227. sqlalchemy/testing/fixtures/mypy.py +332 -0
  228. sqlalchemy/testing/fixtures/orm.py +227 -0
  229. sqlalchemy/testing/fixtures/sql.py +482 -0
  230. sqlalchemy/testing/pickleable.py +155 -0
  231. sqlalchemy/testing/plugin/__init__.py +6 -0
  232. sqlalchemy/testing/plugin/bootstrap.py +51 -0
  233. sqlalchemy/testing/plugin/plugin_base.py +828 -0
  234. sqlalchemy/testing/plugin/pytestplugin.py +892 -0
  235. sqlalchemy/testing/profiling.py +329 -0
  236. sqlalchemy/testing/provision.py +603 -0
  237. sqlalchemy/testing/requirements.py +1945 -0
  238. sqlalchemy/testing/schema.py +198 -0
  239. sqlalchemy/testing/suite/__init__.py +19 -0
  240. sqlalchemy/testing/suite/test_cte.py +237 -0
  241. sqlalchemy/testing/suite/test_ddl.py +389 -0
  242. sqlalchemy/testing/suite/test_deprecations.py +153 -0
  243. sqlalchemy/testing/suite/test_dialect.py +776 -0
  244. sqlalchemy/testing/suite/test_insert.py +630 -0
  245. sqlalchemy/testing/suite/test_reflection.py +3557 -0
  246. sqlalchemy/testing/suite/test_results.py +504 -0
  247. sqlalchemy/testing/suite/test_rowcount.py +258 -0
  248. sqlalchemy/testing/suite/test_select.py +2010 -0
  249. sqlalchemy/testing/suite/test_sequence.py +317 -0
  250. sqlalchemy/testing/suite/test_types.py +2147 -0
  251. sqlalchemy/testing/suite/test_unicode_ddl.py +189 -0
  252. sqlalchemy/testing/suite/test_update_delete.py +139 -0
  253. sqlalchemy/testing/util.py +535 -0
  254. sqlalchemy/testing/warnings.py +52 -0
  255. sqlalchemy/types.py +74 -0
  256. sqlalchemy/util/__init__.py +162 -0
  257. sqlalchemy/util/_collections.py +712 -0
  258. sqlalchemy/util/_concurrency_py3k.py +288 -0
  259. sqlalchemy/util/_has_cy.py +40 -0
  260. sqlalchemy/util/_py_collections.py +541 -0
  261. sqlalchemy/util/compat.py +421 -0
  262. sqlalchemy/util/concurrency.py +110 -0
  263. sqlalchemy/util/deprecations.py +401 -0
  264. sqlalchemy/util/langhelpers.py +2203 -0
  265. sqlalchemy/util/preloaded.py +150 -0
  266. sqlalchemy/util/queue.py +322 -0
  267. sqlalchemy/util/tool_support.py +201 -0
  268. sqlalchemy/util/topological.py +120 -0
  269. sqlalchemy/util/typing.py +734 -0
  270. sqlalchemy-2.0.47.dist-info/METADATA +243 -0
  271. sqlalchemy-2.0.47.dist-info/RECORD +274 -0
  272. sqlalchemy-2.0.47.dist-info/WHEEL +5 -0
  273. sqlalchemy-2.0.47.dist-info/licenses/LICENSE +19 -0
  274. sqlalchemy-2.0.47.dist-info/top_level.txt +1 -0
@@ -0,0 +1,1496 @@
1
+ # orm/interfaces.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
+ """
9
+
10
+ Contains various base classes used throughout the ORM.
11
+
12
+ Defines some key base classes prominent within the internals.
13
+
14
+ This module and the classes within are mostly private, though some attributes
15
+ are exposed when inspecting mappings.
16
+
17
+ """
18
+
19
+ from __future__ import annotations
20
+
21
+ import collections
22
+ import dataclasses
23
+ import typing
24
+ from typing import Any
25
+ from typing import Callable
26
+ from typing import cast
27
+ from typing import ClassVar
28
+ from typing import Dict
29
+ from typing import Generic
30
+ from typing import Iterator
31
+ from typing import List
32
+ from typing import Mapping
33
+ from typing import NamedTuple
34
+ from typing import NoReturn
35
+ from typing import Optional
36
+ from typing import Sequence
37
+ from typing import Set
38
+ from typing import Tuple
39
+ from typing import Type
40
+ from typing import TYPE_CHECKING
41
+ from typing import TypeVar
42
+ from typing import Union
43
+
44
+ from . import exc as orm_exc
45
+ from . import path_registry
46
+ from .base import _MappedAttribute as _MappedAttribute
47
+ from .base import EXT_CONTINUE as EXT_CONTINUE # noqa: F401
48
+ from .base import EXT_SKIP as EXT_SKIP # noqa: F401
49
+ from .base import EXT_STOP as EXT_STOP # noqa: F401
50
+ from .base import InspectionAttr as InspectionAttr # noqa: F401
51
+ from .base import InspectionAttrInfo as InspectionAttrInfo
52
+ from .base import MANYTOMANY as MANYTOMANY # noqa: F401
53
+ from .base import MANYTOONE as MANYTOONE # noqa: F401
54
+ from .base import NO_KEY as NO_KEY # noqa: F401
55
+ from .base import NO_VALUE as NO_VALUE # noqa: F401
56
+ from .base import NotExtension as NotExtension # noqa: F401
57
+ from .base import ONETOMANY as ONETOMANY # noqa: F401
58
+ from .base import RelationshipDirection as RelationshipDirection # noqa: F401
59
+ from .base import SQLORMOperations
60
+ from .. import ColumnElement
61
+ from .. import exc as sa_exc
62
+ from .. import inspection
63
+ from .. import util
64
+ from ..sql import operators
65
+ from ..sql import roles
66
+ from ..sql import visitors
67
+ from ..sql.base import _NoArg
68
+ from ..sql.base import ExecutableOption
69
+ from ..sql.cache_key import HasCacheKey
70
+ from ..sql.operators import ColumnOperators
71
+ from ..sql.schema import Column
72
+ from ..sql.type_api import TypeEngine
73
+ from ..util import warn_deprecated
74
+ from ..util.typing import RODescriptorReference
75
+ from ..util.typing import TypedDict
76
+
77
+ if typing.TYPE_CHECKING:
78
+ from ._typing import _EntityType
79
+ from ._typing import _IdentityKeyType
80
+ from ._typing import _InstanceDict
81
+ from ._typing import _InternalEntityType
82
+ from ._typing import _ORMAdapterProto
83
+ from .attributes import InstrumentedAttribute
84
+ from .base import Mapped
85
+ from .context import _MapperEntity
86
+ from .context import ORMCompileState
87
+ from .context import QueryContext
88
+ from .decl_api import RegistryType
89
+ from .decl_base import _ClassScanMapperConfig
90
+ from .loading import _PopulatorDict
91
+ from .mapper import Mapper
92
+ from .path_registry import AbstractEntityRegistry
93
+ from .query import Query
94
+ from .session import Session
95
+ from .state import InstanceState
96
+ from .strategy_options import _LoadElement
97
+ from .util import AliasedInsp
98
+ from .util import ORMAdapter
99
+ from ..engine.result import Result
100
+ from ..sql._typing import _ColumnExpressionArgument
101
+ from ..sql._typing import _ColumnsClauseArgument
102
+ from ..sql._typing import _DMLColumnArgument
103
+ from ..sql._typing import _InfoType
104
+ from ..sql.operators import OperatorType
105
+ from ..sql.visitors import _TraverseInternalsType
106
+ from ..util.typing import _AnnotationScanType
107
+
108
+ _StrategyKey = Tuple[Any, ...]
109
+
110
+ _T = TypeVar("_T", bound=Any)
111
+ _T_co = TypeVar("_T_co", bound=Any, covariant=True)
112
+
113
+ _TLS = TypeVar("_TLS", bound="Type[LoaderStrategy]")
114
+
115
+
116
+ class ORMStatementRole(roles.StatementRole):
117
+ __slots__ = ()
118
+ _role_name = (
119
+ "Executable SQL or text() construct, including ORM aware objects"
120
+ )
121
+
122
+
123
+ class ORMColumnsClauseRole(
124
+ roles.ColumnsClauseRole, roles.TypedColumnsClauseRole[_T]
125
+ ):
126
+ __slots__ = ()
127
+ _role_name = "ORM mapped entity, aliased entity, or Column expression"
128
+
129
+
130
+ class ORMEntityColumnsClauseRole(ORMColumnsClauseRole[_T]):
131
+ __slots__ = ()
132
+ _role_name = "ORM mapped or aliased entity"
133
+
134
+
135
+ class ORMFromClauseRole(roles.StrictFromClauseRole):
136
+ __slots__ = ()
137
+ _role_name = "ORM mapped entity, aliased entity, or FROM expression"
138
+
139
+
140
+ class ORMColumnDescription(TypedDict):
141
+ name: str
142
+ # TODO: add python_type and sql_type here; combining them
143
+ # into "type" is a bad idea
144
+ type: Union[Type[Any], TypeEngine[Any]]
145
+ aliased: bool
146
+ expr: _ColumnsClauseArgument[Any]
147
+ entity: Optional[_ColumnsClauseArgument[Any]]
148
+
149
+
150
+ class _IntrospectsAnnotations:
151
+ __slots__ = ()
152
+
153
+ @classmethod
154
+ def _mapper_property_name(cls) -> str:
155
+ return cls.__name__
156
+
157
+ def found_in_pep593_annotated(self) -> Any:
158
+ """return a copy of this object to use in declarative when the
159
+ object is found inside of an Annotated object."""
160
+
161
+ raise NotImplementedError(
162
+ f"Use of the {self._mapper_property_name()!r} "
163
+ "construct inside of an Annotated object is not yet supported."
164
+ )
165
+
166
+ def declarative_scan(
167
+ self,
168
+ decl_scan: _ClassScanMapperConfig,
169
+ registry: RegistryType,
170
+ cls: Type[Any],
171
+ originating_module: Optional[str],
172
+ key: str,
173
+ mapped_container: Optional[Type[Mapped[Any]]],
174
+ annotation: Optional[_AnnotationScanType],
175
+ extracted_mapped_annotation: Optional[_AnnotationScanType],
176
+ is_dataclass_field: bool,
177
+ ) -> None:
178
+ """Perform class-specific initialization at early declarative scanning
179
+ time.
180
+
181
+ .. versionadded:: 2.0
182
+
183
+ """
184
+
185
+ def _raise_for_required(self, key: str, cls: Type[Any]) -> NoReturn:
186
+ raise sa_exc.ArgumentError(
187
+ f"Python typing annotation is required for attribute "
188
+ f'"{cls.__name__}.{key}" when primary argument(s) for '
189
+ f'"{self._mapper_property_name()}" '
190
+ "construct are None or not present"
191
+ )
192
+
193
+
194
+ class _AttributeOptions(NamedTuple):
195
+ """define Python-local attribute behavior options common to all
196
+ :class:`.MapperProperty` objects.
197
+
198
+ Currently this includes dataclass-generation arguments.
199
+
200
+ .. versionadded:: 2.0
201
+
202
+ """
203
+
204
+ dataclasses_init: Union[_NoArg, bool]
205
+ dataclasses_repr: Union[_NoArg, bool]
206
+ dataclasses_default: Union[_NoArg, Any]
207
+ dataclasses_default_factory: Union[_NoArg, Callable[[], Any]]
208
+ dataclasses_compare: Union[_NoArg, bool]
209
+ dataclasses_kw_only: Union[_NoArg, bool]
210
+ dataclasses_hash: Union[_NoArg, bool, None]
211
+ dataclasses_dataclass_metadata: Union[_NoArg, Mapping[Any, Any], None]
212
+
213
+ def _as_dataclass_field(self, key: str) -> Any:
214
+ """Return a ``dataclasses.Field`` object given these arguments."""
215
+
216
+ kw: Dict[str, Any] = {}
217
+ if self.dataclasses_default_factory is not _NoArg.NO_ARG:
218
+ kw["default_factory"] = self.dataclasses_default_factory
219
+ if self.dataclasses_default is not _NoArg.NO_ARG:
220
+ kw["default"] = self.dataclasses_default
221
+ if self.dataclasses_init is not _NoArg.NO_ARG:
222
+ kw["init"] = self.dataclasses_init
223
+ if self.dataclasses_repr is not _NoArg.NO_ARG:
224
+ kw["repr"] = self.dataclasses_repr
225
+ if self.dataclasses_compare is not _NoArg.NO_ARG:
226
+ kw["compare"] = self.dataclasses_compare
227
+ if self.dataclasses_kw_only is not _NoArg.NO_ARG:
228
+ kw["kw_only"] = self.dataclasses_kw_only
229
+ if self.dataclasses_hash is not _NoArg.NO_ARG:
230
+ kw["hash"] = self.dataclasses_hash
231
+ if self.dataclasses_dataclass_metadata is not _NoArg.NO_ARG:
232
+ kw["metadata"] = self.dataclasses_dataclass_metadata
233
+
234
+ if "default" in kw and callable(kw["default"]):
235
+ # callable defaults are ambiguous. deprecate them in favour of
236
+ # insert_default or default_factory. #9936
237
+ warn_deprecated(
238
+ f"Callable object passed to the ``default`` parameter for "
239
+ f"attribute {key!r} in a ORM-mapped Dataclasses context is "
240
+ "ambiguous, "
241
+ "and this use will raise an error in a future release. "
242
+ "If this callable is intended to produce Core level INSERT "
243
+ "default values for an underlying ``Column``, use "
244
+ "the ``mapped_column.insert_default`` parameter instead. "
245
+ "To establish this callable as providing a default value "
246
+ "for instances of the dataclass itself, use the "
247
+ "``default_factory`` dataclasses parameter.",
248
+ "2.0",
249
+ )
250
+
251
+ if (
252
+ "init" in kw
253
+ and not kw["init"]
254
+ and "default" in kw
255
+ and not callable(kw["default"]) # ignore callable defaults. #9936
256
+ and "default_factory" not in kw # illegal but let dc.field raise
257
+ ):
258
+ # fix for #9879
259
+ default = kw.pop("default")
260
+ kw["default_factory"] = lambda: default
261
+
262
+ return dataclasses.field(**kw)
263
+
264
+ @classmethod
265
+ def _get_arguments_for_make_dataclass(
266
+ cls,
267
+ key: str,
268
+ annotation: _AnnotationScanType,
269
+ mapped_container: Optional[Any],
270
+ elem: Any,
271
+ ) -> Union[
272
+ Tuple[str, _AnnotationScanType],
273
+ Tuple[str, _AnnotationScanType, dataclasses.Field[Any]],
274
+ ]:
275
+ """given attribute key, annotation, and value from a class, return
276
+ the argument tuple we would pass to dataclasses.make_dataclass()
277
+ for this attribute.
278
+
279
+ """
280
+ if isinstance(elem, _DCAttributeOptions):
281
+ dc_field = elem._attribute_options._as_dataclass_field(key)
282
+
283
+ return (key, annotation, dc_field)
284
+ elif elem is not _NoArg.NO_ARG:
285
+ # why is typing not erroring on this?
286
+ return (key, annotation, elem)
287
+ elif mapped_container is not None:
288
+ # it's Mapped[], but there's no "element", which means declarative
289
+ # did not actually do anything for this field. this shouldn't
290
+ # happen.
291
+ # previously, this would occur because _scan_attributes would
292
+ # skip a field that's on an already mapped superclass, but it
293
+ # would still include it in the annotations, leading
294
+ # to issue #8718
295
+
296
+ assert False, "Mapped[] received without a mapping declaration"
297
+
298
+ else:
299
+ # plain dataclass field, not mapped. Is only possible
300
+ # if __allow_unmapped__ is set up. I can see this mode causing
301
+ # problems...
302
+ return (key, annotation)
303
+
304
+
305
+ _DEFAULT_ATTRIBUTE_OPTIONS = _AttributeOptions(
306
+ _NoArg.NO_ARG,
307
+ _NoArg.NO_ARG,
308
+ _NoArg.NO_ARG,
309
+ _NoArg.NO_ARG,
310
+ _NoArg.NO_ARG,
311
+ _NoArg.NO_ARG,
312
+ _NoArg.NO_ARG,
313
+ _NoArg.NO_ARG,
314
+ )
315
+
316
+ _DEFAULT_READONLY_ATTRIBUTE_OPTIONS = _AttributeOptions(
317
+ False,
318
+ _NoArg.NO_ARG,
319
+ _NoArg.NO_ARG,
320
+ _NoArg.NO_ARG,
321
+ _NoArg.NO_ARG,
322
+ _NoArg.NO_ARG,
323
+ _NoArg.NO_ARG,
324
+ _NoArg.NO_ARG,
325
+ )
326
+
327
+
328
+ class _DCAttributeOptions:
329
+ """mixin for descriptors or configurational objects that include dataclass
330
+ field options.
331
+
332
+ This includes :class:`.MapperProperty`, :class:`._MapsColumn` within
333
+ the ORM, but also includes :class:`.AssociationProxy` within ext.
334
+ Can in theory be used for other descriptors that serve a similar role
335
+ as association proxy. (*maybe* hybrids, not sure yet.)
336
+
337
+ """
338
+
339
+ __slots__ = ()
340
+
341
+ _attribute_options: _AttributeOptions
342
+ """behavioral options for ORM-enabled Python attributes
343
+
344
+ .. versionadded:: 2.0
345
+
346
+ """
347
+
348
+ _has_dataclass_arguments: bool
349
+
350
+
351
+ class _MapsColumns(_DCAttributeOptions, _MappedAttribute[_T]):
352
+ """interface for declarative-capable construct that delivers one or more
353
+ Column objects to the declarative process to be part of a Table.
354
+ """
355
+
356
+ __slots__ = ()
357
+
358
+ @property
359
+ def mapper_property_to_assign(self) -> Optional[MapperProperty[_T]]:
360
+ """return a MapperProperty to be assigned to the declarative mapping"""
361
+ raise NotImplementedError()
362
+
363
+ @property
364
+ def columns_to_assign(self) -> List[Tuple[Column[_T], int]]:
365
+ """A list of Column objects that should be declaratively added to the
366
+ new Table object.
367
+
368
+ """
369
+ raise NotImplementedError()
370
+
371
+
372
+ # NOTE: MapperProperty needs to extend _MappedAttribute so that declarative
373
+ # typing works, i.e. "Mapped[A] = relationship()". This introduces an
374
+ # inconvenience which is that all the MapperProperty objects are treated
375
+ # as descriptors by typing tools, which are misled by this as assignment /
376
+ # access to a descriptor attribute wants to move through __get__.
377
+ # Therefore, references to MapperProperty as an instance variable, such
378
+ # as in PropComparator, may have some special typing workarounds such as the
379
+ # use of sqlalchemy.util.typing.DescriptorReference to avoid mis-interpretation
380
+ # by typing tools
381
+ @inspection._self_inspects
382
+ class MapperProperty(
383
+ HasCacheKey,
384
+ _DCAttributeOptions,
385
+ _MappedAttribute[_T],
386
+ InspectionAttrInfo,
387
+ util.MemoizedSlots,
388
+ ):
389
+ """Represent a particular class attribute mapped by :class:`_orm.Mapper`.
390
+
391
+ The most common occurrences of :class:`.MapperProperty` are the
392
+ mapped :class:`_schema.Column`, which is represented in a mapping as
393
+ an instance of :class:`.ColumnProperty`,
394
+ and a reference to another class produced by :func:`_orm.relationship`,
395
+ represented in the mapping as an instance of
396
+ :class:`.Relationship`.
397
+
398
+ """
399
+
400
+ __slots__ = (
401
+ "_configure_started",
402
+ "_configure_finished",
403
+ "_attribute_options",
404
+ "_has_dataclass_arguments",
405
+ "parent",
406
+ "key",
407
+ "info",
408
+ "doc",
409
+ )
410
+
411
+ _cache_key_traversal: _TraverseInternalsType = [
412
+ ("parent", visitors.ExtendedInternalTraversal.dp_has_cache_key),
413
+ ("key", visitors.ExtendedInternalTraversal.dp_string),
414
+ ]
415
+
416
+ if not TYPE_CHECKING:
417
+ cascade = None
418
+
419
+ is_property = True
420
+ """Part of the InspectionAttr interface; states this object is a
421
+ mapper property.
422
+
423
+ """
424
+
425
+ comparator: PropComparator[_T]
426
+ """The :class:`_orm.PropComparator` instance that implements SQL
427
+ expression construction on behalf of this mapped attribute."""
428
+
429
+ key: str
430
+ """name of class attribute"""
431
+
432
+ parent: Mapper[Any]
433
+ """the :class:`.Mapper` managing this property."""
434
+
435
+ _is_relationship = False
436
+
437
+ _links_to_entity: bool
438
+ """True if this MapperProperty refers to a mapped entity.
439
+
440
+ Should only be True for Relationship, False for all others.
441
+
442
+ """
443
+
444
+ doc: Optional[str]
445
+ """optional documentation string"""
446
+
447
+ info: _InfoType
448
+ """Info dictionary associated with the object, allowing user-defined
449
+ data to be associated with this :class:`.InspectionAttr`.
450
+
451
+ The dictionary is generated when first accessed. Alternatively,
452
+ it can be specified as a constructor argument to the
453
+ :func:`.column_property`, :func:`_orm.relationship`, or :func:`.composite`
454
+ functions.
455
+
456
+ .. seealso::
457
+
458
+ :attr:`.QueryableAttribute.info`
459
+
460
+ :attr:`.SchemaItem.info`
461
+
462
+ """
463
+
464
+ def _memoized_attr_info(self) -> _InfoType:
465
+ """Info dictionary associated with the object, allowing user-defined
466
+ data to be associated with this :class:`.InspectionAttr`.
467
+
468
+ The dictionary is generated when first accessed. Alternatively,
469
+ it can be specified as a constructor argument to the
470
+ :func:`.column_property`, :func:`_orm.relationship`, or
471
+ :func:`.composite`
472
+ functions.
473
+
474
+ .. seealso::
475
+
476
+ :attr:`.QueryableAttribute.info`
477
+
478
+ :attr:`.SchemaItem.info`
479
+
480
+ """
481
+ return {}
482
+
483
+ def setup(
484
+ self,
485
+ context: ORMCompileState,
486
+ query_entity: _MapperEntity,
487
+ path: AbstractEntityRegistry,
488
+ adapter: Optional[ORMAdapter],
489
+ **kwargs: Any,
490
+ ) -> None:
491
+ """Called by Query for the purposes of constructing a SQL statement.
492
+
493
+ Each MapperProperty associated with the target mapper processes the
494
+ statement referenced by the query context, adding columns and/or
495
+ criterion as appropriate.
496
+
497
+ """
498
+
499
+ def create_row_processor(
500
+ self,
501
+ context: ORMCompileState,
502
+ query_entity: _MapperEntity,
503
+ path: AbstractEntityRegistry,
504
+ mapper: Mapper[Any],
505
+ result: Result[Any],
506
+ adapter: Optional[ORMAdapter],
507
+ populators: _PopulatorDict,
508
+ ) -> None:
509
+ """Produce row processing functions and append to the given
510
+ set of populators lists.
511
+
512
+ """
513
+
514
+ def cascade_iterator(
515
+ self,
516
+ type_: str,
517
+ state: InstanceState[Any],
518
+ dict_: _InstanceDict,
519
+ visited_states: Set[InstanceState[Any]],
520
+ halt_on: Optional[Callable[[InstanceState[Any]], bool]] = None,
521
+ ) -> Iterator[
522
+ Tuple[object, Mapper[Any], InstanceState[Any], _InstanceDict]
523
+ ]:
524
+ """Iterate through instances related to the given instance for
525
+ a particular 'cascade', starting with this MapperProperty.
526
+
527
+ Return an iterator3-tuples (instance, mapper, state).
528
+
529
+ Note that the 'cascade' collection on this MapperProperty is
530
+ checked first for the given type before cascade_iterator is called.
531
+
532
+ This method typically only applies to Relationship.
533
+
534
+ """
535
+
536
+ return iter(())
537
+
538
+ def set_parent(self, parent: Mapper[Any], init: bool) -> None:
539
+ """Set the parent mapper that references this MapperProperty.
540
+
541
+ This method is overridden by some subclasses to perform extra
542
+ setup when the mapper is first known.
543
+
544
+ """
545
+ self.parent = parent
546
+
547
+ def instrument_class(self, mapper: Mapper[Any]) -> None:
548
+ """Hook called by the Mapper to the property to initiate
549
+ instrumentation of the class attribute managed by this
550
+ MapperProperty.
551
+
552
+ The MapperProperty here will typically call out to the
553
+ attributes module to set up an InstrumentedAttribute.
554
+
555
+ This step is the first of two steps to set up an InstrumentedAttribute,
556
+ and is called early in the mapper setup process.
557
+
558
+ The second step is typically the init_class_attribute step,
559
+ called from StrategizedProperty via the post_instrument_class()
560
+ hook. This step assigns additional state to the InstrumentedAttribute
561
+ (specifically the "impl") which has been determined after the
562
+ MapperProperty has determined what kind of persistence
563
+ management it needs to do (e.g. scalar, object, collection, etc).
564
+
565
+ """
566
+
567
+ def __init__(
568
+ self,
569
+ attribute_options: Optional[_AttributeOptions] = None,
570
+ _assume_readonly_dc_attributes: bool = False,
571
+ ) -> None:
572
+ self._configure_started = False
573
+ self._configure_finished = False
574
+
575
+ if _assume_readonly_dc_attributes:
576
+ default_attrs = _DEFAULT_READONLY_ATTRIBUTE_OPTIONS
577
+ else:
578
+ default_attrs = _DEFAULT_ATTRIBUTE_OPTIONS
579
+
580
+ if attribute_options and attribute_options != default_attrs:
581
+ self._has_dataclass_arguments = True
582
+ self._attribute_options = attribute_options
583
+ else:
584
+ self._has_dataclass_arguments = False
585
+ self._attribute_options = default_attrs
586
+
587
+ def init(self) -> None:
588
+ """Called after all mappers are created to assemble
589
+ relationships between mappers and perform other post-mapper-creation
590
+ initialization steps.
591
+
592
+
593
+ """
594
+ self._configure_started = True
595
+ self.do_init()
596
+ self._configure_finished = True
597
+
598
+ @property
599
+ def class_attribute(self) -> InstrumentedAttribute[_T]:
600
+ """Return the class-bound descriptor corresponding to this
601
+ :class:`.MapperProperty`.
602
+
603
+ This is basically a ``getattr()`` call::
604
+
605
+ return getattr(self.parent.class_, self.key)
606
+
607
+ I.e. if this :class:`.MapperProperty` were named ``addresses``,
608
+ and the class to which it is mapped is ``User``, this sequence
609
+ is possible::
610
+
611
+ >>> from sqlalchemy import inspect
612
+ >>> mapper = inspect(User)
613
+ >>> addresses_property = mapper.attrs.addresses
614
+ >>> addresses_property.class_attribute is User.addresses
615
+ True
616
+ >>> User.addresses.property is addresses_property
617
+ True
618
+
619
+
620
+ """
621
+
622
+ return getattr(self.parent.class_, self.key) # type: ignore
623
+
624
+ def do_init(self) -> None:
625
+ """Perform subclass-specific initialization post-mapper-creation
626
+ steps.
627
+
628
+ This is a template method called by the ``MapperProperty``
629
+ object's init() method.
630
+
631
+ """
632
+
633
+ def post_instrument_class(self, mapper: Mapper[Any]) -> None:
634
+ """Perform instrumentation adjustments that need to occur
635
+ after init() has completed.
636
+
637
+ The given Mapper is the Mapper invoking the operation, which
638
+ may not be the same Mapper as self.parent in an inheritance
639
+ scenario; however, Mapper will always at least be a sub-mapper of
640
+ self.parent.
641
+
642
+ This method is typically used by StrategizedProperty, which delegates
643
+ it to LoaderStrategy.init_class_attribute() to perform final setup
644
+ on the class-bound InstrumentedAttribute.
645
+
646
+ """
647
+
648
+ def merge(
649
+ self,
650
+ session: Session,
651
+ source_state: InstanceState[Any],
652
+ source_dict: _InstanceDict,
653
+ dest_state: InstanceState[Any],
654
+ dest_dict: _InstanceDict,
655
+ load: bool,
656
+ _recursive: Dict[Any, object],
657
+ _resolve_conflict_map: Dict[_IdentityKeyType[Any], object],
658
+ ) -> None:
659
+ """Merge the attribute represented by this ``MapperProperty``
660
+ from source to destination object.
661
+
662
+ """
663
+
664
+ def __repr__(self) -> str:
665
+ return "<%s at 0x%x; %s>" % (
666
+ self.__class__.__name__,
667
+ id(self),
668
+ getattr(self, "key", "no key"),
669
+ )
670
+
671
+
672
+ @inspection._self_inspects
673
+ class PropComparator(SQLORMOperations[_T_co], Generic[_T_co], ColumnOperators):
674
+ r"""Defines SQL operations for ORM mapped attributes.
675
+
676
+ SQLAlchemy allows for operators to
677
+ be redefined at both the Core and ORM level. :class:`.PropComparator`
678
+ is the base class of operator redefinition for ORM-level operations,
679
+ including those of :class:`.ColumnProperty`,
680
+ :class:`.Relationship`, and :class:`.Composite`.
681
+
682
+ User-defined subclasses of :class:`.PropComparator` may be created. The
683
+ built-in Python comparison and math operator methods, such as
684
+ :meth:`.operators.ColumnOperators.__eq__`,
685
+ :meth:`.operators.ColumnOperators.__lt__`, and
686
+ :meth:`.operators.ColumnOperators.__add__`, can be overridden to provide
687
+ new operator behavior. The custom :class:`.PropComparator` is passed to
688
+ the :class:`.MapperProperty` instance via the ``comparator_factory``
689
+ argument. In each case,
690
+ the appropriate subclass of :class:`.PropComparator` should be used::
691
+
692
+ # definition of custom PropComparator subclasses
693
+
694
+ from sqlalchemy.orm.properties import (
695
+ ColumnProperty,
696
+ Composite,
697
+ Relationship,
698
+ )
699
+
700
+
701
+ class MyColumnComparator(ColumnProperty.Comparator):
702
+ def __eq__(self, other):
703
+ return self.__clause_element__() == other
704
+
705
+
706
+ class MyRelationshipComparator(Relationship.Comparator):
707
+ def any(self, expression):
708
+ "define the 'any' operation"
709
+ # ...
710
+
711
+
712
+ class MyCompositeComparator(Composite.Comparator):
713
+ def __gt__(self, other):
714
+ "redefine the 'greater than' operation"
715
+
716
+ return sql.and_(
717
+ *[
718
+ a > b
719
+ for a, b in zip(
720
+ self.__clause_element__().clauses,
721
+ other.__composite_values__(),
722
+ )
723
+ ]
724
+ )
725
+
726
+
727
+ # application of custom PropComparator subclasses
728
+
729
+ from sqlalchemy.orm import column_property, relationship, composite
730
+ from sqlalchemy import Column, String
731
+
732
+
733
+ class SomeMappedClass(Base):
734
+ some_column = column_property(
735
+ Column("some_column", String),
736
+ comparator_factory=MyColumnComparator,
737
+ )
738
+
739
+ some_relationship = relationship(
740
+ SomeOtherClass, comparator_factory=MyRelationshipComparator
741
+ )
742
+
743
+ some_composite = composite(
744
+ Column("a", String),
745
+ Column("b", String),
746
+ comparator_factory=MyCompositeComparator,
747
+ )
748
+
749
+ Note that for column-level operator redefinition, it's usually
750
+ simpler to define the operators at the Core level, using the
751
+ :attr:`.TypeEngine.comparator_factory` attribute. See
752
+ :ref:`types_operators` for more detail.
753
+
754
+ .. seealso::
755
+
756
+ :class:`.ColumnProperty.Comparator`
757
+
758
+ :class:`.Relationship.Comparator`
759
+
760
+ :class:`.Composite.Comparator`
761
+
762
+ :class:`.ColumnOperators`
763
+
764
+ :ref:`types_operators`
765
+
766
+ :attr:`.TypeEngine.comparator_factory`
767
+
768
+ """
769
+
770
+ __slots__ = "prop", "_parententity", "_adapt_to_entity"
771
+
772
+ __visit_name__ = "orm_prop_comparator"
773
+
774
+ _parententity: _InternalEntityType[Any]
775
+ _adapt_to_entity: Optional[AliasedInsp[Any]]
776
+ prop: RODescriptorReference[MapperProperty[_T_co]]
777
+
778
+ def __init__(
779
+ self,
780
+ prop: MapperProperty[_T],
781
+ parentmapper: _InternalEntityType[Any],
782
+ adapt_to_entity: Optional[AliasedInsp[Any]] = None,
783
+ ):
784
+ self.prop = prop
785
+ self._parententity = adapt_to_entity or parentmapper
786
+ self._adapt_to_entity = adapt_to_entity
787
+
788
+ @util.non_memoized_property
789
+ def property(self) -> MapperProperty[_T_co]:
790
+ """Return the :class:`.MapperProperty` associated with this
791
+ :class:`.PropComparator`.
792
+
793
+
794
+ Return values here will commonly be instances of
795
+ :class:`.ColumnProperty` or :class:`.Relationship`.
796
+
797
+
798
+ """
799
+ return self.prop
800
+
801
+ def __clause_element__(self) -> roles.ColumnsClauseRole:
802
+ raise NotImplementedError("%r" % self)
803
+
804
+ def _bulk_update_tuples(
805
+ self, value: Any
806
+ ) -> Sequence[Tuple[_DMLColumnArgument, Any]]:
807
+ """Receive a SQL expression that represents a value in the SET
808
+ clause of an UPDATE statement.
809
+
810
+ Return a tuple that can be passed to a :class:`_expression.Update`
811
+ construct.
812
+
813
+ """
814
+
815
+ return [(cast("_DMLColumnArgument", self.__clause_element__()), value)]
816
+
817
+ def adapt_to_entity(
818
+ self, adapt_to_entity: AliasedInsp[Any]
819
+ ) -> PropComparator[_T_co]:
820
+ """Return a copy of this PropComparator which will use the given
821
+ :class:`.AliasedInsp` to produce corresponding expressions.
822
+ """
823
+ return self.__class__(self.prop, self._parententity, adapt_to_entity)
824
+
825
+ @util.ro_non_memoized_property
826
+ def _parentmapper(self) -> Mapper[Any]:
827
+ """legacy; this is renamed to _parententity to be
828
+ compatible with QueryableAttribute."""
829
+ return self._parententity.mapper
830
+
831
+ def _criterion_exists(
832
+ self,
833
+ criterion: Optional[_ColumnExpressionArgument[bool]] = None,
834
+ **kwargs: Any,
835
+ ) -> ColumnElement[Any]:
836
+ return self.prop.comparator._criterion_exists(criterion, **kwargs)
837
+
838
+ @util.ro_non_memoized_property
839
+ def adapter(self) -> Optional[_ORMAdapterProto]:
840
+ """Produce a callable that adapts column expressions
841
+ to suit an aliased version of this comparator.
842
+
843
+ """
844
+ if self._adapt_to_entity is None:
845
+ return None
846
+ else:
847
+ return self._adapt_to_entity._orm_adapt_element
848
+
849
+ @util.ro_non_memoized_property
850
+ def info(self) -> _InfoType:
851
+ return self.prop.info
852
+
853
+ @staticmethod
854
+ def _any_op(a: Any, b: Any, **kwargs: Any) -> Any:
855
+ return a.any(b, **kwargs)
856
+
857
+ @staticmethod
858
+ def _has_op(left: Any, other: Any, **kwargs: Any) -> Any:
859
+ return left.has(other, **kwargs)
860
+
861
+ @staticmethod
862
+ def _of_type_op(a: Any, class_: Any) -> Any:
863
+ return a.of_type(class_)
864
+
865
+ any_op = cast(operators.OperatorType, _any_op)
866
+ has_op = cast(operators.OperatorType, _has_op)
867
+ of_type_op = cast(operators.OperatorType, _of_type_op)
868
+
869
+ if typing.TYPE_CHECKING:
870
+
871
+ def operate(
872
+ self, op: OperatorType, *other: Any, **kwargs: Any
873
+ ) -> ColumnElement[Any]: ...
874
+
875
+ def reverse_operate(
876
+ self, op: OperatorType, other: Any, **kwargs: Any
877
+ ) -> ColumnElement[Any]: ...
878
+
879
+ def of_type(self, class_: _EntityType[Any]) -> PropComparator[_T_co]:
880
+ r"""Redefine this object in terms of a polymorphic subclass,
881
+ :func:`_orm.with_polymorphic` construct, or :func:`_orm.aliased`
882
+ construct.
883
+
884
+ Returns a new PropComparator from which further criterion can be
885
+ evaluated.
886
+
887
+ e.g.::
888
+
889
+ query.join(Company.employees.of_type(Engineer)).filter(
890
+ Engineer.name == "foo"
891
+ )
892
+
893
+ :param \class_: a class or mapper indicating that criterion will be
894
+ against this specific subclass.
895
+
896
+ .. seealso::
897
+
898
+ :ref:`orm_queryguide_joining_relationships_aliased` - in the
899
+ :ref:`queryguide_toplevel`
900
+
901
+ :ref:`inheritance_of_type`
902
+
903
+ """
904
+
905
+ return self.operate(PropComparator.of_type_op, class_) # type: ignore
906
+
907
+ def and_(
908
+ self, *criteria: _ColumnExpressionArgument[bool]
909
+ ) -> PropComparator[bool]:
910
+ """Add additional criteria to the ON clause that's represented by this
911
+ relationship attribute.
912
+
913
+ E.g.::
914
+
915
+
916
+ stmt = select(User).join(
917
+ User.addresses.and_(Address.email_address != "foo")
918
+ )
919
+
920
+ stmt = select(User).options(
921
+ joinedload(User.addresses.and_(Address.email_address != "foo"))
922
+ )
923
+
924
+ .. versionadded:: 1.4
925
+
926
+ .. seealso::
927
+
928
+ :ref:`orm_queryguide_join_on_augmented`
929
+
930
+ :ref:`loader_option_criteria`
931
+
932
+ :func:`.with_loader_criteria`
933
+
934
+ """
935
+ return self.operate(operators.and_, *criteria) # type: ignore
936
+
937
+ def any(
938
+ self,
939
+ criterion: Optional[_ColumnExpressionArgument[bool]] = None,
940
+ **kwargs: Any,
941
+ ) -> ColumnElement[bool]:
942
+ r"""Return a SQL expression representing true if this element
943
+ references a member which meets the given criterion.
944
+
945
+ The usual implementation of ``any()`` is
946
+ :meth:`.Relationship.Comparator.any`.
947
+
948
+ :param criterion: an optional ClauseElement formulated against the
949
+ member class' table or attributes.
950
+
951
+ :param \**kwargs: key/value pairs corresponding to member class
952
+ attribute names which will be compared via equality to the
953
+ corresponding values.
954
+
955
+ """
956
+
957
+ return self.operate(PropComparator.any_op, criterion, **kwargs)
958
+
959
+ def has(
960
+ self,
961
+ criterion: Optional[_ColumnExpressionArgument[bool]] = None,
962
+ **kwargs: Any,
963
+ ) -> ColumnElement[bool]:
964
+ r"""Return a SQL expression representing true if this element
965
+ references a member which meets the given criterion.
966
+
967
+ The usual implementation of ``has()`` is
968
+ :meth:`.Relationship.Comparator.has`.
969
+
970
+ :param criterion: an optional ClauseElement formulated against the
971
+ member class' table or attributes.
972
+
973
+ :param \**kwargs: key/value pairs corresponding to member class
974
+ attribute names which will be compared via equality to the
975
+ corresponding values.
976
+
977
+ """
978
+
979
+ return self.operate(PropComparator.has_op, criterion, **kwargs)
980
+
981
+
982
+ class StrategizedProperty(MapperProperty[_T]):
983
+ """A MapperProperty which uses selectable strategies to affect
984
+ loading behavior.
985
+
986
+ There is a single strategy selected by default. Alternate
987
+ strategies can be selected at Query time through the usage of
988
+ ``StrategizedOption`` objects via the Query.options() method.
989
+
990
+ The mechanics of StrategizedProperty are used for every Query
991
+ invocation for every mapped attribute participating in that Query,
992
+ to determine first how the attribute will be rendered in SQL
993
+ and secondly how the attribute will retrieve a value from a result
994
+ row and apply it to a mapped object. The routines here are very
995
+ performance-critical.
996
+
997
+ """
998
+
999
+ __slots__ = (
1000
+ "_strategies",
1001
+ "strategy",
1002
+ "_wildcard_token",
1003
+ "_default_path_loader_key",
1004
+ "strategy_key",
1005
+ )
1006
+ inherit_cache = True
1007
+ strategy_wildcard_key: ClassVar[str]
1008
+
1009
+ strategy_key: _StrategyKey
1010
+
1011
+ _strategies: Dict[_StrategyKey, LoaderStrategy]
1012
+
1013
+ def _memoized_attr__wildcard_token(self) -> Tuple[str]:
1014
+ return (
1015
+ f"{self.strategy_wildcard_key}:{path_registry._WILDCARD_TOKEN}",
1016
+ )
1017
+
1018
+ def _memoized_attr__default_path_loader_key(
1019
+ self,
1020
+ ) -> Tuple[str, Tuple[str]]:
1021
+ return (
1022
+ "loader",
1023
+ (f"{self.strategy_wildcard_key}:{path_registry._DEFAULT_TOKEN}",),
1024
+ )
1025
+
1026
+ def _get_context_loader(
1027
+ self, context: ORMCompileState, path: AbstractEntityRegistry
1028
+ ) -> Optional[_LoadElement]:
1029
+ load: Optional[_LoadElement] = None
1030
+
1031
+ search_path = path[self]
1032
+
1033
+ # search among: exact match, "attr.*", "default" strategy
1034
+ # if any.
1035
+ for path_key in (
1036
+ search_path._loader_key,
1037
+ search_path._wildcard_path_loader_key,
1038
+ search_path._default_path_loader_key,
1039
+ ):
1040
+ if path_key in context.attributes:
1041
+ load = context.attributes[path_key]
1042
+ break
1043
+
1044
+ # note that if strategy_options.Load is placing non-actionable
1045
+ # objects in the context like defaultload(), we would
1046
+ # need to continue the loop here if we got such an
1047
+ # option as below.
1048
+ # if load.strategy or load.local_opts:
1049
+ # break
1050
+
1051
+ return load
1052
+
1053
+ def _get_strategy(self, key: _StrategyKey) -> LoaderStrategy:
1054
+ try:
1055
+ return self._strategies[key]
1056
+ except KeyError:
1057
+ pass
1058
+
1059
+ # run outside to prevent transfer of exception context
1060
+ cls = self._strategy_lookup(self, *key)
1061
+ # this previously was setting self._strategies[cls], that's
1062
+ # a bad idea; should use strategy key at all times because every
1063
+ # strategy has multiple keys at this point
1064
+ self._strategies[key] = strategy = cls(self, key)
1065
+ return strategy
1066
+
1067
+ def setup(
1068
+ self,
1069
+ context: ORMCompileState,
1070
+ query_entity: _MapperEntity,
1071
+ path: AbstractEntityRegistry,
1072
+ adapter: Optional[ORMAdapter],
1073
+ **kwargs: Any,
1074
+ ) -> None:
1075
+ loader = self._get_context_loader(context, path)
1076
+ if loader and loader.strategy:
1077
+ strat = self._get_strategy(loader.strategy)
1078
+ else:
1079
+ strat = self.strategy
1080
+ strat.setup_query(
1081
+ context, query_entity, path, loader, adapter, **kwargs
1082
+ )
1083
+
1084
+ def create_row_processor(
1085
+ self,
1086
+ context: ORMCompileState,
1087
+ query_entity: _MapperEntity,
1088
+ path: AbstractEntityRegistry,
1089
+ mapper: Mapper[Any],
1090
+ result: Result[Any],
1091
+ adapter: Optional[ORMAdapter],
1092
+ populators: _PopulatorDict,
1093
+ ) -> None:
1094
+ loader = self._get_context_loader(context, path)
1095
+ if loader and loader.strategy:
1096
+ strat = self._get_strategy(loader.strategy)
1097
+ else:
1098
+ strat = self.strategy
1099
+ strat.create_row_processor(
1100
+ context,
1101
+ query_entity,
1102
+ path,
1103
+ loader,
1104
+ mapper,
1105
+ result,
1106
+ adapter,
1107
+ populators,
1108
+ )
1109
+
1110
+ def do_init(self) -> None:
1111
+ self._strategies = {}
1112
+ self.strategy = self._get_strategy(self.strategy_key)
1113
+
1114
+ def post_instrument_class(self, mapper: Mapper[Any]) -> None:
1115
+ if (
1116
+ not self.parent.non_primary
1117
+ and not mapper.class_manager._attr_has_impl(self.key)
1118
+ ):
1119
+ self.strategy.init_class_attribute(mapper)
1120
+
1121
+ _all_strategies: collections.defaultdict[
1122
+ Type[MapperProperty[Any]], Dict[_StrategyKey, Type[LoaderStrategy]]
1123
+ ] = collections.defaultdict(dict)
1124
+
1125
+ @classmethod
1126
+ def strategy_for(cls, **kw: Any) -> Callable[[_TLS], _TLS]:
1127
+ def decorate(dec_cls: _TLS) -> _TLS:
1128
+ # ensure each subclass of the strategy has its
1129
+ # own _strategy_keys collection
1130
+ if "_strategy_keys" not in dec_cls.__dict__:
1131
+ dec_cls._strategy_keys = []
1132
+ key = tuple(sorted(kw.items()))
1133
+ cls._all_strategies[cls][key] = dec_cls
1134
+ dec_cls._strategy_keys.append(key)
1135
+ return dec_cls
1136
+
1137
+ return decorate
1138
+
1139
+ @classmethod
1140
+ def _strategy_lookup(
1141
+ cls, requesting_property: MapperProperty[Any], *key: Any
1142
+ ) -> Type[LoaderStrategy]:
1143
+ requesting_property.parent._with_polymorphic_mappers
1144
+
1145
+ for prop_cls in cls.__mro__:
1146
+ if prop_cls in cls._all_strategies:
1147
+ if TYPE_CHECKING:
1148
+ assert issubclass(prop_cls, MapperProperty)
1149
+ strategies = cls._all_strategies[prop_cls]
1150
+ try:
1151
+ return strategies[key]
1152
+ except KeyError:
1153
+ pass
1154
+
1155
+ for property_type, strats in cls._all_strategies.items():
1156
+ if key in strats:
1157
+ intended_property_type = property_type
1158
+ actual_strategy = strats[key]
1159
+ break
1160
+ else:
1161
+ intended_property_type = None
1162
+ actual_strategy = None
1163
+
1164
+ raise orm_exc.LoaderStrategyException(
1165
+ cls,
1166
+ requesting_property,
1167
+ intended_property_type,
1168
+ actual_strategy,
1169
+ key,
1170
+ )
1171
+
1172
+
1173
+ class ORMOption(ExecutableOption):
1174
+ """Base class for option objects that are passed to ORM queries.
1175
+
1176
+ These options may be consumed by :meth:`.Query.options`,
1177
+ :meth:`.Select.options`, or in a more general sense by any
1178
+ :meth:`.Executable.options` method. They are interpreted at
1179
+ statement compile time or execution time in modern use. The
1180
+ deprecated :class:`.MapperOption` is consumed at ORM query construction
1181
+ time.
1182
+
1183
+ .. versionadded:: 1.4
1184
+
1185
+ """
1186
+
1187
+ __slots__ = ()
1188
+
1189
+ _is_legacy_option = False
1190
+
1191
+ propagate_to_loaders = False
1192
+ """if True, indicate this option should be carried along
1193
+ to "secondary" SELECT statements that occur for relationship
1194
+ lazy loaders as well as attribute load / refresh operations.
1195
+
1196
+ """
1197
+
1198
+ _is_core = False
1199
+
1200
+ _is_user_defined = False
1201
+
1202
+ _is_compile_state = False
1203
+
1204
+ _is_criteria_option = False
1205
+
1206
+ _is_strategy_option = False
1207
+
1208
+ def _adapt_cached_option_to_uncached_option(
1209
+ self, context: QueryContext, uncached_opt: ORMOption
1210
+ ) -> ORMOption:
1211
+ """adapt this option to the "uncached" version of itself in a
1212
+ loader strategy context.
1213
+
1214
+ given "self" which is an option from a cached query, as well as the
1215
+ corresponding option from the uncached version of the same query,
1216
+ return the option we should use in a new query, in the context of a
1217
+ loader strategy being asked to load related rows on behalf of that
1218
+ cached query, which is assumed to be building a new query based on
1219
+ entities passed to us from the cached query.
1220
+
1221
+ Currently this routine chooses between "self" and "uncached" without
1222
+ manufacturing anything new. If the option is itself a loader strategy
1223
+ option which has a path, that path needs to match to the entities being
1224
+ passed to us by the cached query, so the :class:`_orm.Load` subclass
1225
+ overrides this to return "self". For all other options, we return the
1226
+ uncached form which may have changing state, such as a
1227
+ with_loader_criteria() option which will very often have new state.
1228
+
1229
+ This routine could in the future involve
1230
+ generating a new option based on both inputs if use cases arise,
1231
+ such as if with_loader_criteria() needed to match up to
1232
+ ``AliasedClass`` instances given in the parent query.
1233
+
1234
+ However, longer term it might be better to restructure things such that
1235
+ ``AliasedClass`` entities are always matched up on their cache key,
1236
+ instead of identity, in things like paths and such, so that this whole
1237
+ issue of "the uncached option does not match the entities" goes away.
1238
+ However this would make ``PathRegistry`` more complicated and difficult
1239
+ to debug as well as potentially less performant in that it would be
1240
+ hashing enormous cache keys rather than a simple AliasedInsp. UNLESS,
1241
+ we could get cache keys overall to be reliably hashed into something
1242
+ like an md5 key.
1243
+
1244
+ .. versionadded:: 1.4.41
1245
+
1246
+ """
1247
+ if uncached_opt is not None:
1248
+ return uncached_opt
1249
+ else:
1250
+ return self
1251
+
1252
+
1253
+ class CompileStateOption(HasCacheKey, ORMOption):
1254
+ """base for :class:`.ORMOption` classes that affect the compilation of
1255
+ a SQL query and therefore need to be part of the cache key.
1256
+
1257
+ .. note:: :class:`.CompileStateOption` is generally non-public and
1258
+ should not be used as a base class for user-defined options; instead,
1259
+ use :class:`.UserDefinedOption`, which is easier to use as it does not
1260
+ interact with ORM compilation internals or caching.
1261
+
1262
+ :class:`.CompileStateOption` defines an internal attribute
1263
+ ``_is_compile_state=True`` which has the effect of the ORM compilation
1264
+ routines for SELECT and other statements will call upon these options when
1265
+ a SQL string is being compiled. As such, these classes implement
1266
+ :class:`.HasCacheKey` and need to provide robust ``_cache_key_traversal``
1267
+ structures.
1268
+
1269
+ The :class:`.CompileStateOption` class is used to implement the ORM
1270
+ :class:`.LoaderOption` and :class:`.CriteriaOption` classes.
1271
+
1272
+ .. versionadded:: 1.4.28
1273
+
1274
+
1275
+ """
1276
+
1277
+ __slots__ = ()
1278
+
1279
+ _is_compile_state = True
1280
+
1281
+ def process_compile_state(self, compile_state: ORMCompileState) -> None:
1282
+ """Apply a modification to a given :class:`.ORMCompileState`.
1283
+
1284
+ This method is part of the implementation of a particular
1285
+ :class:`.CompileStateOption` and is only invoked internally
1286
+ when an ORM query is compiled.
1287
+
1288
+ """
1289
+
1290
+ def process_compile_state_replaced_entities(
1291
+ self,
1292
+ compile_state: ORMCompileState,
1293
+ mapper_entities: Sequence[_MapperEntity],
1294
+ ) -> None:
1295
+ """Apply a modification to a given :class:`.ORMCompileState`,
1296
+ given entities that were replaced by with_only_columns() or
1297
+ with_entities().
1298
+
1299
+ This method is part of the implementation of a particular
1300
+ :class:`.CompileStateOption` and is only invoked internally
1301
+ when an ORM query is compiled.
1302
+
1303
+ .. versionadded:: 1.4.19
1304
+
1305
+ """
1306
+
1307
+
1308
+ class LoaderOption(CompileStateOption):
1309
+ """Describe a loader modification to an ORM statement at compilation time.
1310
+
1311
+ .. versionadded:: 1.4
1312
+
1313
+ """
1314
+
1315
+ __slots__ = ()
1316
+
1317
+ def process_compile_state_replaced_entities(
1318
+ self,
1319
+ compile_state: ORMCompileState,
1320
+ mapper_entities: Sequence[_MapperEntity],
1321
+ ) -> None:
1322
+ self.process_compile_state(compile_state)
1323
+
1324
+
1325
+ class CriteriaOption(CompileStateOption):
1326
+ """Describe a WHERE criteria modification to an ORM statement at
1327
+ compilation time.
1328
+
1329
+ .. versionadded:: 1.4
1330
+
1331
+ """
1332
+
1333
+ __slots__ = ()
1334
+
1335
+ _is_criteria_option = True
1336
+
1337
+ def get_global_criteria(self, attributes: Dict[str, Any]) -> None:
1338
+ """update additional entity criteria options in the given
1339
+ attributes dictionary.
1340
+
1341
+ """
1342
+
1343
+
1344
+ class UserDefinedOption(ORMOption):
1345
+ """Base class for a user-defined option that can be consumed from the
1346
+ :meth:`.SessionEvents.do_orm_execute` event hook.
1347
+
1348
+ """
1349
+
1350
+ __slots__ = ("payload",)
1351
+
1352
+ _is_legacy_option = False
1353
+
1354
+ _is_user_defined = True
1355
+
1356
+ propagate_to_loaders = False
1357
+ """if True, indicate this option should be carried along
1358
+ to "secondary" Query objects produced during lazy loads
1359
+ or refresh operations.
1360
+
1361
+ """
1362
+
1363
+ def __init__(self, payload: Optional[Any] = None):
1364
+ self.payload = payload
1365
+
1366
+
1367
+ @util.deprecated_cls(
1368
+ "1.4",
1369
+ "The :class:`.MapperOption class is deprecated and will be removed "
1370
+ "in a future release. For "
1371
+ "modifications to queries on a per-execution basis, use the "
1372
+ ":class:`.UserDefinedOption` class to establish state within a "
1373
+ ":class:`.Query` or other Core statement, then use the "
1374
+ ":meth:`.SessionEvents.before_orm_execute` hook to consume them.",
1375
+ constructor=None,
1376
+ )
1377
+ class MapperOption(ORMOption):
1378
+ """Describe a modification to a Query"""
1379
+
1380
+ __slots__ = ()
1381
+
1382
+ _is_legacy_option = True
1383
+
1384
+ propagate_to_loaders = False
1385
+ """if True, indicate this option should be carried along
1386
+ to "secondary" Query objects produced during lazy loads
1387
+ or refresh operations.
1388
+
1389
+ """
1390
+
1391
+ def process_query(self, query: Query[Any]) -> None:
1392
+ """Apply a modification to the given :class:`_query.Query`."""
1393
+
1394
+ def process_query_conditionally(self, query: Query[Any]) -> None:
1395
+ """same as process_query(), except that this option may not
1396
+ apply to the given query.
1397
+
1398
+ This is typically applied during a lazy load or scalar refresh
1399
+ operation to propagate options stated in the original Query to the
1400
+ new Query being used for the load. It occurs for those options that
1401
+ specify propagate_to_loaders=True.
1402
+
1403
+ """
1404
+
1405
+ self.process_query(query)
1406
+
1407
+
1408
+ class LoaderStrategy:
1409
+ """Describe the loading behavior of a StrategizedProperty object.
1410
+
1411
+ The ``LoaderStrategy`` interacts with the querying process in three
1412
+ ways:
1413
+
1414
+ * it controls the configuration of the ``InstrumentedAttribute``
1415
+ placed on a class to handle the behavior of the attribute. this
1416
+ may involve setting up class-level callable functions to fire
1417
+ off a select operation when the attribute is first accessed
1418
+ (i.e. a lazy load)
1419
+
1420
+ * it processes the ``QueryContext`` at statement construction time,
1421
+ where it can modify the SQL statement that is being produced.
1422
+ For example, simple column attributes will add their represented
1423
+ column to the list of selected columns, a joined eager loader
1424
+ may establish join clauses to add to the statement.
1425
+
1426
+ * It produces "row processor" functions at result fetching time.
1427
+ These "row processor" functions populate a particular attribute
1428
+ on a particular mapped instance.
1429
+
1430
+ """
1431
+
1432
+ __slots__ = (
1433
+ "parent_property",
1434
+ "is_class_level",
1435
+ "parent",
1436
+ "key",
1437
+ "strategy_key",
1438
+ "strategy_opts",
1439
+ )
1440
+
1441
+ _strategy_keys: ClassVar[List[_StrategyKey]]
1442
+
1443
+ def __init__(
1444
+ self, parent: MapperProperty[Any], strategy_key: _StrategyKey
1445
+ ):
1446
+ self.parent_property = parent
1447
+ self.is_class_level = False
1448
+ self.parent = self.parent_property.parent
1449
+ self.key = self.parent_property.key
1450
+ self.strategy_key = strategy_key
1451
+ self.strategy_opts = dict(strategy_key)
1452
+
1453
+ def init_class_attribute(self, mapper: Mapper[Any]) -> None:
1454
+ pass
1455
+
1456
+ def setup_query(
1457
+ self,
1458
+ compile_state: ORMCompileState,
1459
+ query_entity: _MapperEntity,
1460
+ path: AbstractEntityRegistry,
1461
+ loadopt: Optional[_LoadElement],
1462
+ adapter: Optional[ORMAdapter],
1463
+ **kwargs: Any,
1464
+ ) -> None:
1465
+ """Establish column and other state for a given QueryContext.
1466
+
1467
+ This method fulfills the contract specified by MapperProperty.setup().
1468
+
1469
+ StrategizedProperty delegates its setup() method
1470
+ directly to this method.
1471
+
1472
+ """
1473
+
1474
+ def create_row_processor(
1475
+ self,
1476
+ context: ORMCompileState,
1477
+ query_entity: _MapperEntity,
1478
+ path: AbstractEntityRegistry,
1479
+ loadopt: Optional[_LoadElement],
1480
+ mapper: Mapper[Any],
1481
+ result: Result[Any],
1482
+ adapter: Optional[ORMAdapter],
1483
+ populators: _PopulatorDict,
1484
+ ) -> None:
1485
+ """Establish row processing functions for a given QueryContext.
1486
+
1487
+ This method fulfills the contract specified by
1488
+ MapperProperty.create_row_processor().
1489
+
1490
+ StrategizedProperty delegates its create_row_processor() method
1491
+ directly to this method.
1492
+
1493
+ """
1494
+
1495
+ def __str__(self) -> str:
1496
+ return str(self.parent_property)