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,300 @@
1
+ # orm/dynamic.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
+ """Dynamic collection API.
10
+
11
+ Dynamic collections act like Query() objects for read operations and support
12
+ basic add/delete mutation.
13
+
14
+ .. legacy:: the "dynamic" loader is a legacy feature, superseded by the
15
+ "write_only" loader.
16
+
17
+
18
+ """
19
+
20
+ from __future__ import annotations
21
+
22
+ from typing import Any
23
+ from typing import Iterable
24
+ from typing import Iterator
25
+ from typing import List
26
+ from typing import Optional
27
+ from typing import Tuple
28
+ from typing import Type
29
+ from typing import TYPE_CHECKING
30
+ from typing import TypeVar
31
+ from typing import Union
32
+
33
+ from . import attributes
34
+ from . import exc as orm_exc
35
+ from . import relationships
36
+ from . import util as orm_util
37
+ from .base import PassiveFlag
38
+ from .query import Query
39
+ from .session import object_session
40
+ from .writeonly import AbstractCollectionWriter
41
+ from .writeonly import WriteOnlyAttributeImpl
42
+ from .writeonly import WriteOnlyHistory
43
+ from .writeonly import WriteOnlyLoader
44
+ from .. import util
45
+ from ..engine import result
46
+
47
+
48
+ if TYPE_CHECKING:
49
+ from . import QueryableAttribute
50
+ from .mapper import Mapper
51
+ from .relationships import _RelationshipOrderByArg
52
+ from .session import Session
53
+ from .state import InstanceState
54
+ from .util import AliasedClass
55
+ from ..event import _Dispatch
56
+ from ..sql.elements import ColumnElement
57
+
58
+ _T = TypeVar("_T", bound=Any)
59
+
60
+
61
+ class DynamicCollectionHistory(WriteOnlyHistory[_T]):
62
+ def __init__(
63
+ self,
64
+ attr: DynamicAttributeImpl,
65
+ state: InstanceState[_T],
66
+ passive: PassiveFlag,
67
+ apply_to: Optional[DynamicCollectionHistory[_T]] = None,
68
+ ) -> None:
69
+ if apply_to:
70
+ coll = AppenderQuery(attr, state).autoflush(False)
71
+ self.unchanged_items = util.OrderedIdentitySet(coll)
72
+ self.added_items = apply_to.added_items
73
+ self.deleted_items = apply_to.deleted_items
74
+ self._reconcile_collection = True
75
+ else:
76
+ self.deleted_items = util.OrderedIdentitySet()
77
+ self.added_items = util.OrderedIdentitySet()
78
+ self.unchanged_items = util.OrderedIdentitySet()
79
+ self._reconcile_collection = False
80
+
81
+
82
+ class DynamicAttributeImpl(WriteOnlyAttributeImpl):
83
+ _supports_dynamic_iteration = True
84
+ collection_history_cls = DynamicCollectionHistory[Any]
85
+ query_class: Type[AppenderMixin[Any]] # type: ignore[assignment]
86
+
87
+ def __init__(
88
+ self,
89
+ class_: Union[Type[Any], AliasedClass[Any]],
90
+ key: str,
91
+ dispatch: _Dispatch[QueryableAttribute[Any]],
92
+ target_mapper: Mapper[_T],
93
+ order_by: _RelationshipOrderByArg,
94
+ query_class: Optional[Type[AppenderMixin[_T]]] = None,
95
+ **kw: Any,
96
+ ) -> None:
97
+ attributes.AttributeImpl.__init__(
98
+ self, class_, key, None, dispatch, **kw
99
+ )
100
+ self.target_mapper = target_mapper
101
+ if order_by:
102
+ self.order_by = tuple(order_by)
103
+ if not query_class:
104
+ self.query_class = AppenderQuery
105
+ elif AppenderMixin in query_class.mro():
106
+ self.query_class = query_class
107
+ else:
108
+ self.query_class = mixin_user_query(query_class)
109
+
110
+
111
+ @relationships.RelationshipProperty.strategy_for(lazy="dynamic")
112
+ class DynaLoader(WriteOnlyLoader):
113
+ impl_class = DynamicAttributeImpl
114
+
115
+
116
+ class AppenderMixin(AbstractCollectionWriter[_T]):
117
+ """A mixin that expects to be mixing in a Query class with
118
+ AbstractAppender.
119
+
120
+
121
+ """
122
+
123
+ query_class: Optional[Type[Query[_T]]] = None
124
+ _order_by_clauses: Tuple[ColumnElement[Any], ...]
125
+
126
+ def __init__(
127
+ self, attr: DynamicAttributeImpl, state: InstanceState[_T]
128
+ ) -> None:
129
+ Query.__init__(
130
+ self, # type: ignore[arg-type]
131
+ attr.target_mapper,
132
+ None,
133
+ )
134
+ super().__init__(attr, state)
135
+
136
+ @property
137
+ def session(self) -> Optional[Session]:
138
+ sess = object_session(self.instance)
139
+ if sess is not None and sess.autoflush and self.instance in sess:
140
+ sess.flush()
141
+ if not orm_util.has_identity(self.instance):
142
+ return None
143
+ else:
144
+ return sess
145
+
146
+ @session.setter
147
+ def session(self, session: Session) -> None:
148
+ self.sess = session
149
+
150
+ def _iter(self) -> Union[result.ScalarResult[_T], result.Result[_T]]:
151
+ sess = self.session
152
+ if sess is None:
153
+ state = attributes.instance_state(self.instance)
154
+ if state.detached:
155
+ util.warn(
156
+ "Instance %s is detached, dynamic relationship cannot "
157
+ "return a correct result. This warning will become "
158
+ "a DetachedInstanceError in a future release."
159
+ % (orm_util.state_str(state))
160
+ )
161
+
162
+ return result.IteratorResult(
163
+ result.SimpleResultMetaData([self.attr.class_.__name__]),
164
+ iter(
165
+ self.attr._get_collection_history(
166
+ attributes.instance_state(self.instance),
167
+ PassiveFlag.PASSIVE_NO_INITIALIZE,
168
+ ).added_items
169
+ ),
170
+ _source_supports_scalars=True,
171
+ ).scalars()
172
+ else:
173
+ return self._generate(sess)._iter()
174
+
175
+ if TYPE_CHECKING:
176
+
177
+ def __iter__(self) -> Iterator[_T]: ...
178
+
179
+ def __getitem__(self, index: Any) -> Union[_T, List[_T]]:
180
+ sess = self.session
181
+ if sess is None:
182
+ return self.attr._get_collection_history(
183
+ attributes.instance_state(self.instance),
184
+ PassiveFlag.PASSIVE_NO_INITIALIZE,
185
+ ).indexed(index)
186
+ else:
187
+ return self._generate(sess).__getitem__(index) # type: ignore[no-any-return] # noqa: E501
188
+
189
+ def count(self) -> int:
190
+ sess = self.session
191
+ if sess is None:
192
+ return len(
193
+ self.attr._get_collection_history(
194
+ attributes.instance_state(self.instance),
195
+ PassiveFlag.PASSIVE_NO_INITIALIZE,
196
+ ).added_items
197
+ )
198
+ else:
199
+ return self._generate(sess).count()
200
+
201
+ def _generate(
202
+ self,
203
+ sess: Optional[Session] = None,
204
+ ) -> Query[_T]:
205
+ # note we're returning an entirely new Query class instance
206
+ # here without any assignment capabilities; the class of this
207
+ # query is determined by the session.
208
+ instance = self.instance
209
+ if sess is None:
210
+ sess = object_session(instance)
211
+ if sess is None:
212
+ raise orm_exc.DetachedInstanceError(
213
+ "Parent instance %s is not bound to a Session, and no "
214
+ "contextual session is established; lazy load operation "
215
+ "of attribute '%s' cannot proceed"
216
+ % (orm_util.instance_str(instance), self.attr.key)
217
+ )
218
+
219
+ if self.query_class:
220
+ query = self.query_class(self.attr.target_mapper, session=sess)
221
+ else:
222
+ query = sess.query(self.attr.target_mapper)
223
+
224
+ query._where_criteria = self._where_criteria
225
+ query._from_obj = self._from_obj
226
+ query._order_by_clauses = self._order_by_clauses
227
+
228
+ return query
229
+
230
+ def add_all(self, iterator: Iterable[_T]) -> None:
231
+ """Add an iterable of items to this :class:`_orm.AppenderQuery`.
232
+
233
+ The given items will be persisted to the database in terms of
234
+ the parent instance's collection on the next flush.
235
+
236
+ This method is provided to assist in delivering forwards-compatibility
237
+ with the :class:`_orm.WriteOnlyCollection` collection class.
238
+
239
+ .. versionadded:: 2.0
240
+
241
+ """
242
+ self._add_all_impl(iterator)
243
+
244
+ def add(self, item: _T) -> None:
245
+ """Add an item to this :class:`_orm.AppenderQuery`.
246
+
247
+ The given item will be persisted to the database in terms of
248
+ the parent instance's collection on the next flush.
249
+
250
+ This method is provided to assist in delivering forwards-compatibility
251
+ with the :class:`_orm.WriteOnlyCollection` collection class.
252
+
253
+ .. versionadded:: 2.0
254
+
255
+ """
256
+ self._add_all_impl([item])
257
+
258
+ def extend(self, iterator: Iterable[_T]) -> None:
259
+ """Add an iterable of items to this :class:`_orm.AppenderQuery`.
260
+
261
+ The given items will be persisted to the database in terms of
262
+ the parent instance's collection on the next flush.
263
+
264
+ """
265
+ self._add_all_impl(iterator)
266
+
267
+ def append(self, item: _T) -> None:
268
+ """Append an item to this :class:`_orm.AppenderQuery`.
269
+
270
+ The given item will be persisted to the database in terms of
271
+ the parent instance's collection on the next flush.
272
+
273
+ """
274
+ self._add_all_impl([item])
275
+
276
+ def remove(self, item: _T) -> None:
277
+ """Remove an item from this :class:`_orm.AppenderQuery`.
278
+
279
+ The given item will be removed from the parent instance's collection on
280
+ the next flush.
281
+
282
+ """
283
+ self._remove_impl(item)
284
+
285
+
286
+ class AppenderQuery(AppenderMixin[_T], Query[_T]): # type: ignore[misc]
287
+ """A dynamic query that supports basic collection storage operations.
288
+
289
+ Methods on :class:`.AppenderQuery` include all methods of
290
+ :class:`_orm.Query`, plus additional methods used for collection
291
+ persistence.
292
+
293
+
294
+ """
295
+
296
+
297
+ def mixin_user_query(cls: Any) -> type[AppenderMixin[Any]]:
298
+ """Return a new class with AppenderQuery functionality layered over."""
299
+ name = "Appender" + cls.__name__
300
+ return type(name, (AppenderMixin, cls), {"query_class": cls})
@@ -0,0 +1,379 @@
1
+ # orm/evaluator.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
+ """Evaluation functions used **INTERNALLY** by ORM DML use cases.
10
+
11
+
12
+ This module is **private, for internal use by SQLAlchemy**.
13
+
14
+ .. versionchanged:: 2.0.4 renamed ``EvaluatorCompiler`` to
15
+ ``_EvaluatorCompiler``.
16
+
17
+ """
18
+
19
+
20
+ from __future__ import annotations
21
+
22
+ from typing import Type
23
+
24
+ from . import exc as orm_exc
25
+ from .base import LoaderCallableStatus
26
+ from .base import PassiveFlag
27
+ from .. import exc
28
+ from .. import inspect
29
+ from ..sql import and_
30
+ from ..sql import operators
31
+ from ..sql.sqltypes import Concatenable
32
+ from ..sql.sqltypes import Integer
33
+ from ..sql.sqltypes import Numeric
34
+ from ..util import warn_deprecated
35
+
36
+
37
+ class UnevaluatableError(exc.InvalidRequestError):
38
+ pass
39
+
40
+
41
+ class _NoObject(operators.ColumnOperators):
42
+ def operate(self, *arg, **kw):
43
+ return None
44
+
45
+ def reverse_operate(self, *arg, **kw):
46
+ return None
47
+
48
+
49
+ class _ExpiredObject(operators.ColumnOperators):
50
+ def operate(self, *arg, **kw):
51
+ return self
52
+
53
+ def reverse_operate(self, *arg, **kw):
54
+ return self
55
+
56
+
57
+ _NO_OBJECT = _NoObject()
58
+ _EXPIRED_OBJECT = _ExpiredObject()
59
+
60
+
61
+ class _EvaluatorCompiler:
62
+ def __init__(self, target_cls=None):
63
+ self.target_cls = target_cls
64
+
65
+ def process(self, clause, *clauses):
66
+ if clauses:
67
+ clause = and_(clause, *clauses)
68
+
69
+ meth = getattr(self, f"visit_{clause.__visit_name__}", None)
70
+ if not meth:
71
+ raise UnevaluatableError(
72
+ f"Cannot evaluate {type(clause).__name__}"
73
+ )
74
+ return meth(clause)
75
+
76
+ def visit_grouping(self, clause):
77
+ return self.process(clause.element)
78
+
79
+ def visit_null(self, clause):
80
+ return lambda obj: None
81
+
82
+ def visit_false(self, clause):
83
+ return lambda obj: False
84
+
85
+ def visit_true(self, clause):
86
+ return lambda obj: True
87
+
88
+ def visit_column(self, clause):
89
+ try:
90
+ parentmapper = clause._annotations["parentmapper"]
91
+ except KeyError as ke:
92
+ raise UnevaluatableError(
93
+ f"Cannot evaluate column: {clause}"
94
+ ) from ke
95
+
96
+ if self.target_cls and not issubclass(
97
+ self.target_cls, parentmapper.class_
98
+ ):
99
+ raise UnevaluatableError(
100
+ "Can't evaluate criteria against "
101
+ f"alternate class {parentmapper.class_}"
102
+ )
103
+
104
+ parentmapper._check_configure()
105
+
106
+ # we'd like to use "proxy_key" annotation to get the "key", however
107
+ # in relationship primaryjoin cases proxy_key is sometimes deannotated
108
+ # and sometimes apparently not present in the first place (?).
109
+ # While I can stop it from being deannotated (though need to see if
110
+ # this breaks other things), not sure right now about cases where it's
111
+ # not there in the first place. can fix at some later point.
112
+ # key = clause._annotations["proxy_key"]
113
+
114
+ # for now, use the old way
115
+ try:
116
+ key = parentmapper._columntoproperty[clause].key
117
+ except orm_exc.UnmappedColumnError as err:
118
+ raise UnevaluatableError(
119
+ f"Cannot evaluate expression: {err}"
120
+ ) from err
121
+
122
+ # note this used to fall back to a simple `getattr(obj, key)` evaluator
123
+ # if impl was None; as of #8656, we ensure mappers are configured
124
+ # so that impl is available
125
+ impl = parentmapper.class_manager[key].impl
126
+
127
+ def get_corresponding_attr(obj):
128
+ if obj is None:
129
+ return _NO_OBJECT
130
+ state = inspect(obj)
131
+ dict_ = state.dict
132
+
133
+ value = impl.get(
134
+ state, dict_, passive=PassiveFlag.PASSIVE_NO_FETCH
135
+ )
136
+ if value is LoaderCallableStatus.PASSIVE_NO_RESULT:
137
+ return _EXPIRED_OBJECT
138
+ return value
139
+
140
+ return get_corresponding_attr
141
+
142
+ def visit_tuple(self, clause):
143
+ return self.visit_clauselist(clause)
144
+
145
+ def visit_expression_clauselist(self, clause):
146
+ return self.visit_clauselist(clause)
147
+
148
+ def visit_clauselist(self, clause):
149
+ evaluators = [self.process(clause) for clause in clause.clauses]
150
+
151
+ dispatch = (
152
+ f"visit_{clause.operator.__name__.rstrip('_')}_clauselist_op"
153
+ )
154
+ meth = getattr(self, dispatch, None)
155
+ if meth:
156
+ return meth(clause.operator, evaluators, clause)
157
+ else:
158
+ raise UnevaluatableError(
159
+ f"Cannot evaluate clauselist with operator {clause.operator}"
160
+ )
161
+
162
+ def visit_binary(self, clause):
163
+ eval_left = self.process(clause.left)
164
+ eval_right = self.process(clause.right)
165
+
166
+ dispatch = f"visit_{clause.operator.__name__.rstrip('_')}_binary_op"
167
+ meth = getattr(self, dispatch, None)
168
+ if meth:
169
+ return meth(clause.operator, eval_left, eval_right, clause)
170
+ else:
171
+ raise UnevaluatableError(
172
+ f"Cannot evaluate {type(clause).__name__} with "
173
+ f"operator {clause.operator}"
174
+ )
175
+
176
+ def visit_or_clauselist_op(self, operator, evaluators, clause):
177
+ def evaluate(obj):
178
+ has_null = False
179
+ for sub_evaluate in evaluators:
180
+ value = sub_evaluate(obj)
181
+ if value is _EXPIRED_OBJECT:
182
+ return _EXPIRED_OBJECT
183
+ elif value:
184
+ return True
185
+ has_null = has_null or value is None
186
+ if has_null:
187
+ return None
188
+ return False
189
+
190
+ return evaluate
191
+
192
+ def visit_and_clauselist_op(self, operator, evaluators, clause):
193
+ def evaluate(obj):
194
+ for sub_evaluate in evaluators:
195
+ value = sub_evaluate(obj)
196
+ if value is _EXPIRED_OBJECT:
197
+ return _EXPIRED_OBJECT
198
+
199
+ if not value:
200
+ if value is None or value is _NO_OBJECT:
201
+ return None
202
+ return False
203
+ return True
204
+
205
+ return evaluate
206
+
207
+ def visit_comma_op_clauselist_op(self, operator, evaluators, clause):
208
+ def evaluate(obj):
209
+ values = []
210
+ for sub_evaluate in evaluators:
211
+ value = sub_evaluate(obj)
212
+ if value is _EXPIRED_OBJECT:
213
+ return _EXPIRED_OBJECT
214
+ elif value is None or value is _NO_OBJECT:
215
+ return None
216
+ values.append(value)
217
+ return tuple(values)
218
+
219
+ return evaluate
220
+
221
+ def visit_custom_op_binary_op(
222
+ self, operator, eval_left, eval_right, clause
223
+ ):
224
+ if operator.python_impl:
225
+ return self._straight_evaluate(
226
+ operator, eval_left, eval_right, clause
227
+ )
228
+ else:
229
+ raise UnevaluatableError(
230
+ f"Custom operator {operator.opstring!r} can't be evaluated "
231
+ "in Python unless it specifies a callable using "
232
+ "`.python_impl`."
233
+ )
234
+
235
+ def visit_is_binary_op(self, operator, eval_left, eval_right, clause):
236
+ def evaluate(obj):
237
+ left_val = eval_left(obj)
238
+ right_val = eval_right(obj)
239
+ if left_val is _EXPIRED_OBJECT or right_val is _EXPIRED_OBJECT:
240
+ return _EXPIRED_OBJECT
241
+ return left_val == right_val
242
+
243
+ return evaluate
244
+
245
+ def visit_is_not_binary_op(self, operator, eval_left, eval_right, clause):
246
+ def evaluate(obj):
247
+ left_val = eval_left(obj)
248
+ right_val = eval_right(obj)
249
+ if left_val is _EXPIRED_OBJECT or right_val is _EXPIRED_OBJECT:
250
+ return _EXPIRED_OBJECT
251
+ return left_val != right_val
252
+
253
+ return evaluate
254
+
255
+ def _straight_evaluate(self, operator, eval_left, eval_right, clause):
256
+ def evaluate(obj):
257
+ left_val = eval_left(obj)
258
+ right_val = eval_right(obj)
259
+ if left_val is _EXPIRED_OBJECT or right_val is _EXPIRED_OBJECT:
260
+ return _EXPIRED_OBJECT
261
+ elif left_val is None or right_val is None:
262
+ return None
263
+
264
+ return operator(eval_left(obj), eval_right(obj))
265
+
266
+ return evaluate
267
+
268
+ def _straight_evaluate_numeric_only(
269
+ self, operator, eval_left, eval_right, clause
270
+ ):
271
+ if clause.left.type._type_affinity not in (
272
+ Numeric,
273
+ Integer,
274
+ ) or clause.right.type._type_affinity not in (Numeric, Integer):
275
+ raise UnevaluatableError(
276
+ f'Cannot evaluate math operator "{operator.__name__}" for '
277
+ f"datatypes {clause.left.type}, {clause.right.type}"
278
+ )
279
+
280
+ return self._straight_evaluate(operator, eval_left, eval_right, clause)
281
+
282
+ visit_add_binary_op = _straight_evaluate_numeric_only
283
+ visit_mul_binary_op = _straight_evaluate_numeric_only
284
+ visit_sub_binary_op = _straight_evaluate_numeric_only
285
+ visit_mod_binary_op = _straight_evaluate_numeric_only
286
+ visit_truediv_binary_op = _straight_evaluate_numeric_only
287
+ visit_lt_binary_op = _straight_evaluate
288
+ visit_le_binary_op = _straight_evaluate
289
+ visit_ne_binary_op = _straight_evaluate
290
+ visit_gt_binary_op = _straight_evaluate
291
+ visit_ge_binary_op = _straight_evaluate
292
+ visit_eq_binary_op = _straight_evaluate
293
+
294
+ def visit_in_op_binary_op(self, operator, eval_left, eval_right, clause):
295
+ return self._straight_evaluate(
296
+ lambda a, b: a in b if a is not _NO_OBJECT else None,
297
+ eval_left,
298
+ eval_right,
299
+ clause,
300
+ )
301
+
302
+ def visit_not_in_op_binary_op(
303
+ self, operator, eval_left, eval_right, clause
304
+ ):
305
+ return self._straight_evaluate(
306
+ lambda a, b: a not in b if a is not _NO_OBJECT else None,
307
+ eval_left,
308
+ eval_right,
309
+ clause,
310
+ )
311
+
312
+ def visit_concat_op_binary_op(
313
+ self, operator, eval_left, eval_right, clause
314
+ ):
315
+
316
+ if not issubclass(
317
+ clause.left.type._type_affinity, Concatenable
318
+ ) or not issubclass(clause.right.type._type_affinity, Concatenable):
319
+ raise UnevaluatableError(
320
+ f"Cannot evaluate concatenate operator "
321
+ f'"{operator.__name__}" for '
322
+ f"datatypes {clause.left.type}, {clause.right.type}"
323
+ )
324
+
325
+ return self._straight_evaluate(
326
+ lambda a, b: a + b, eval_left, eval_right, clause
327
+ )
328
+
329
+ def visit_startswith_op_binary_op(
330
+ self, operator, eval_left, eval_right, clause
331
+ ):
332
+ return self._straight_evaluate(
333
+ lambda a, b: a.startswith(b), eval_left, eval_right, clause
334
+ )
335
+
336
+ def visit_endswith_op_binary_op(
337
+ self, operator, eval_left, eval_right, clause
338
+ ):
339
+ return self._straight_evaluate(
340
+ lambda a, b: a.endswith(b), eval_left, eval_right, clause
341
+ )
342
+
343
+ def visit_unary(self, clause):
344
+ eval_inner = self.process(clause.element)
345
+ if clause.operator is operators.inv:
346
+
347
+ def evaluate(obj):
348
+ value = eval_inner(obj)
349
+ if value is _EXPIRED_OBJECT:
350
+ return _EXPIRED_OBJECT
351
+ elif value is None:
352
+ return None
353
+ return not value
354
+
355
+ return evaluate
356
+ raise UnevaluatableError(
357
+ f"Cannot evaluate {type(clause).__name__} "
358
+ f"with operator {clause.operator}"
359
+ )
360
+
361
+ def visit_bindparam(self, clause):
362
+ if clause.callable:
363
+ val = clause.callable()
364
+ else:
365
+ val = clause.value
366
+ return lambda obj: val
367
+
368
+
369
+ def __getattr__(name: str) -> Type[_EvaluatorCompiler]:
370
+ if name == "EvaluatorCompiler":
371
+ warn_deprecated(
372
+ "Direct use of 'EvaluatorCompiler' is not supported, and this "
373
+ "name will be removed in a future release. "
374
+ "'_EvaluatorCompiler' is for internal use only",
375
+ "2.0",
376
+ )
377
+ return _EvaluatorCompiler
378
+ else:
379
+ raise AttributeError(f"module {__name__!r} has no attribute {name!r}")