SQLAlchemy 2.1.0b2__cp313-cp313t-win_arm64.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (270) hide show
  1. sqlalchemy/__init__.py +298 -0
  2. sqlalchemy/connectors/__init__.py +18 -0
  3. sqlalchemy/connectors/aioodbc.py +171 -0
  4. sqlalchemy/connectors/asyncio.py +476 -0
  5. sqlalchemy/connectors/pyodbc.py +250 -0
  6. sqlalchemy/dialects/__init__.py +62 -0
  7. sqlalchemy/dialects/_typing.py +30 -0
  8. sqlalchemy/dialects/mssql/__init__.py +89 -0
  9. sqlalchemy/dialects/mssql/aioodbc.py +63 -0
  10. sqlalchemy/dialects/mssql/base.py +4166 -0
  11. sqlalchemy/dialects/mssql/information_schema.py +285 -0
  12. sqlalchemy/dialects/mssql/json.py +140 -0
  13. sqlalchemy/dialects/mssql/mssqlpython.py +220 -0
  14. sqlalchemy/dialects/mssql/provision.py +196 -0
  15. sqlalchemy/dialects/mssql/pymssql.py +126 -0
  16. sqlalchemy/dialects/mssql/pyodbc.py +698 -0
  17. sqlalchemy/dialects/mysql/__init__.py +106 -0
  18. sqlalchemy/dialects/mysql/_mariadb_shim.py +312 -0
  19. sqlalchemy/dialects/mysql/aiomysql.py +226 -0
  20. sqlalchemy/dialects/mysql/asyncmy.py +214 -0
  21. sqlalchemy/dialects/mysql/base.py +3877 -0
  22. sqlalchemy/dialects/mysql/cymysql.py +106 -0
  23. sqlalchemy/dialects/mysql/dml.py +279 -0
  24. sqlalchemy/dialects/mysql/enumerated.py +277 -0
  25. sqlalchemy/dialects/mysql/expression.py +146 -0
  26. sqlalchemy/dialects/mysql/json.py +92 -0
  27. sqlalchemy/dialects/mysql/mariadb.py +67 -0
  28. sqlalchemy/dialects/mysql/mariadbconnector.py +330 -0
  29. sqlalchemy/dialects/mysql/mysqlconnector.py +296 -0
  30. sqlalchemy/dialects/mysql/mysqldb.py +312 -0
  31. sqlalchemy/dialects/mysql/provision.py +153 -0
  32. sqlalchemy/dialects/mysql/pymysql.py +157 -0
  33. sqlalchemy/dialects/mysql/pyodbc.py +156 -0
  34. sqlalchemy/dialects/mysql/reflection.py +724 -0
  35. sqlalchemy/dialects/mysql/reserved_words.py +570 -0
  36. sqlalchemy/dialects/mysql/types.py +845 -0
  37. sqlalchemy/dialects/oracle/__init__.py +85 -0
  38. sqlalchemy/dialects/oracle/base.py +3977 -0
  39. sqlalchemy/dialects/oracle/cx_oracle.py +1601 -0
  40. sqlalchemy/dialects/oracle/dictionary.py +507 -0
  41. sqlalchemy/dialects/oracle/json.py +158 -0
  42. sqlalchemy/dialects/oracle/oracledb.py +909 -0
  43. sqlalchemy/dialects/oracle/provision.py +288 -0
  44. sqlalchemy/dialects/oracle/types.py +367 -0
  45. sqlalchemy/dialects/oracle/vector.py +368 -0
  46. sqlalchemy/dialects/postgresql/__init__.py +171 -0
  47. sqlalchemy/dialects/postgresql/_psycopg_common.py +229 -0
  48. sqlalchemy/dialects/postgresql/array.py +534 -0
  49. sqlalchemy/dialects/postgresql/asyncpg.py +1323 -0
  50. sqlalchemy/dialects/postgresql/base.py +5789 -0
  51. sqlalchemy/dialects/postgresql/bitstring.py +327 -0
  52. sqlalchemy/dialects/postgresql/dml.py +360 -0
  53. sqlalchemy/dialects/postgresql/ext.py +593 -0
  54. sqlalchemy/dialects/postgresql/hstore.py +423 -0
  55. sqlalchemy/dialects/postgresql/json.py +408 -0
  56. sqlalchemy/dialects/postgresql/named_types.py +521 -0
  57. sqlalchemy/dialects/postgresql/operators.py +130 -0
  58. sqlalchemy/dialects/postgresql/pg8000.py +670 -0
  59. sqlalchemy/dialects/postgresql/pg_catalog.py +344 -0
  60. sqlalchemy/dialects/postgresql/provision.py +184 -0
  61. sqlalchemy/dialects/postgresql/psycopg.py +799 -0
  62. sqlalchemy/dialects/postgresql/psycopg2.py +860 -0
  63. sqlalchemy/dialects/postgresql/psycopg2cffi.py +61 -0
  64. sqlalchemy/dialects/postgresql/ranges.py +1002 -0
  65. sqlalchemy/dialects/postgresql/types.py +388 -0
  66. sqlalchemy/dialects/sqlite/__init__.py +57 -0
  67. sqlalchemy/dialects/sqlite/aiosqlite.py +321 -0
  68. sqlalchemy/dialects/sqlite/base.py +3063 -0
  69. sqlalchemy/dialects/sqlite/dml.py +279 -0
  70. sqlalchemy/dialects/sqlite/json.py +100 -0
  71. sqlalchemy/dialects/sqlite/provision.py +229 -0
  72. sqlalchemy/dialects/sqlite/pysqlcipher.py +161 -0
  73. sqlalchemy/dialects/sqlite/pysqlite.py +754 -0
  74. sqlalchemy/dialects/type_migration_guidelines.txt +145 -0
  75. sqlalchemy/engine/__init__.py +62 -0
  76. sqlalchemy/engine/_processors_cy.cp313t-win_arm64.pyd +0 -0
  77. sqlalchemy/engine/_processors_cy.py +92 -0
  78. sqlalchemy/engine/_result_cy.cp313t-win_arm64.pyd +0 -0
  79. sqlalchemy/engine/_result_cy.py +633 -0
  80. sqlalchemy/engine/_row_cy.cp313t-win_arm64.pyd +0 -0
  81. sqlalchemy/engine/_row_cy.py +232 -0
  82. sqlalchemy/engine/_util_cy.cp313t-win_arm64.pyd +0 -0
  83. sqlalchemy/engine/_util_cy.py +136 -0
  84. sqlalchemy/engine/base.py +3354 -0
  85. sqlalchemy/engine/characteristics.py +155 -0
  86. sqlalchemy/engine/create.py +877 -0
  87. sqlalchemy/engine/cursor.py +2421 -0
  88. sqlalchemy/engine/default.py +2402 -0
  89. sqlalchemy/engine/events.py +965 -0
  90. sqlalchemy/engine/interfaces.py +3495 -0
  91. sqlalchemy/engine/mock.py +134 -0
  92. sqlalchemy/engine/processors.py +82 -0
  93. sqlalchemy/engine/reflection.py +2100 -0
  94. sqlalchemy/engine/result.py +1966 -0
  95. sqlalchemy/engine/row.py +397 -0
  96. sqlalchemy/engine/strategies.py +16 -0
  97. sqlalchemy/engine/url.py +922 -0
  98. sqlalchemy/engine/util.py +156 -0
  99. sqlalchemy/event/__init__.py +26 -0
  100. sqlalchemy/event/api.py +220 -0
  101. sqlalchemy/event/attr.py +674 -0
  102. sqlalchemy/event/base.py +472 -0
  103. sqlalchemy/event/legacy.py +258 -0
  104. sqlalchemy/event/registry.py +390 -0
  105. sqlalchemy/events.py +17 -0
  106. sqlalchemy/exc.py +922 -0
  107. sqlalchemy/ext/__init__.py +11 -0
  108. sqlalchemy/ext/associationproxy.py +2072 -0
  109. sqlalchemy/ext/asyncio/__init__.py +29 -0
  110. sqlalchemy/ext/asyncio/base.py +281 -0
  111. sqlalchemy/ext/asyncio/engine.py +1487 -0
  112. sqlalchemy/ext/asyncio/exc.py +21 -0
  113. sqlalchemy/ext/asyncio/result.py +994 -0
  114. sqlalchemy/ext/asyncio/scoping.py +1679 -0
  115. sqlalchemy/ext/asyncio/session.py +2007 -0
  116. sqlalchemy/ext/automap.py +1701 -0
  117. sqlalchemy/ext/baked.py +559 -0
  118. sqlalchemy/ext/compiler.py +600 -0
  119. sqlalchemy/ext/declarative/__init__.py +65 -0
  120. sqlalchemy/ext/declarative/extensions.py +560 -0
  121. sqlalchemy/ext/horizontal_shard.py +481 -0
  122. sqlalchemy/ext/hybrid.py +1877 -0
  123. sqlalchemy/ext/indexable.py +364 -0
  124. sqlalchemy/ext/instrumentation.py +450 -0
  125. sqlalchemy/ext/mutable.py +1081 -0
  126. sqlalchemy/ext/orderinglist.py +439 -0
  127. sqlalchemy/ext/serializer.py +185 -0
  128. sqlalchemy/future/__init__.py +16 -0
  129. sqlalchemy/future/engine.py +15 -0
  130. sqlalchemy/inspection.py +174 -0
  131. sqlalchemy/log.py +283 -0
  132. sqlalchemy/orm/__init__.py +176 -0
  133. sqlalchemy/orm/_orm_constructors.py +2694 -0
  134. sqlalchemy/orm/_typing.py +179 -0
  135. sqlalchemy/orm/attributes.py +2868 -0
  136. sqlalchemy/orm/base.py +976 -0
  137. sqlalchemy/orm/bulk_persistence.py +2152 -0
  138. sqlalchemy/orm/clsregistry.py +582 -0
  139. sqlalchemy/orm/collections.py +1568 -0
  140. sqlalchemy/orm/context.py +3471 -0
  141. sqlalchemy/orm/decl_api.py +2280 -0
  142. sqlalchemy/orm/decl_base.py +2309 -0
  143. sqlalchemy/orm/dependency.py +1306 -0
  144. sqlalchemy/orm/descriptor_props.py +1183 -0
  145. sqlalchemy/orm/dynamic.py +307 -0
  146. sqlalchemy/orm/evaluator.py +379 -0
  147. sqlalchemy/orm/events.py +3386 -0
  148. sqlalchemy/orm/exc.py +237 -0
  149. sqlalchemy/orm/identity.py +302 -0
  150. sqlalchemy/orm/instrumentation.py +746 -0
  151. sqlalchemy/orm/interfaces.py +1589 -0
  152. sqlalchemy/orm/loading.py +1684 -0
  153. sqlalchemy/orm/mapped_collection.py +557 -0
  154. sqlalchemy/orm/mapper.py +4411 -0
  155. sqlalchemy/orm/path_registry.py +829 -0
  156. sqlalchemy/orm/persistence.py +1789 -0
  157. sqlalchemy/orm/properties.py +973 -0
  158. sqlalchemy/orm/query.py +3528 -0
  159. sqlalchemy/orm/relationships.py +3570 -0
  160. sqlalchemy/orm/scoping.py +2232 -0
  161. sqlalchemy/orm/session.py +5403 -0
  162. sqlalchemy/orm/state.py +1175 -0
  163. sqlalchemy/orm/state_changes.py +196 -0
  164. sqlalchemy/orm/strategies.py +3492 -0
  165. sqlalchemy/orm/strategy_options.py +2562 -0
  166. sqlalchemy/orm/sync.py +164 -0
  167. sqlalchemy/orm/unitofwork.py +798 -0
  168. sqlalchemy/orm/util.py +2438 -0
  169. sqlalchemy/orm/writeonly.py +694 -0
  170. sqlalchemy/pool/__init__.py +41 -0
  171. sqlalchemy/pool/base.py +1522 -0
  172. sqlalchemy/pool/events.py +375 -0
  173. sqlalchemy/pool/impl.py +582 -0
  174. sqlalchemy/py.typed +0 -0
  175. sqlalchemy/schema.py +74 -0
  176. sqlalchemy/sql/__init__.py +156 -0
  177. sqlalchemy/sql/_annotated_cols.py +397 -0
  178. sqlalchemy/sql/_dml_constructors.py +132 -0
  179. sqlalchemy/sql/_elements_constructors.py +2164 -0
  180. sqlalchemy/sql/_orm_types.py +20 -0
  181. sqlalchemy/sql/_selectable_constructors.py +840 -0
  182. sqlalchemy/sql/_typing.py +487 -0
  183. sqlalchemy/sql/_util_cy.cp313t-win_arm64.pyd +0 -0
  184. sqlalchemy/sql/_util_cy.py +127 -0
  185. sqlalchemy/sql/annotation.py +590 -0
  186. sqlalchemy/sql/base.py +2699 -0
  187. sqlalchemy/sql/cache_key.py +1066 -0
  188. sqlalchemy/sql/coercions.py +1373 -0
  189. sqlalchemy/sql/compiler.py +8327 -0
  190. sqlalchemy/sql/crud.py +1815 -0
  191. sqlalchemy/sql/ddl.py +1928 -0
  192. sqlalchemy/sql/default_comparator.py +654 -0
  193. sqlalchemy/sql/dml.py +1977 -0
  194. sqlalchemy/sql/elements.py +6033 -0
  195. sqlalchemy/sql/events.py +458 -0
  196. sqlalchemy/sql/expression.py +172 -0
  197. sqlalchemy/sql/functions.py +2305 -0
  198. sqlalchemy/sql/lambdas.py +1443 -0
  199. sqlalchemy/sql/naming.py +209 -0
  200. sqlalchemy/sql/operators.py +2897 -0
  201. sqlalchemy/sql/roles.py +332 -0
  202. sqlalchemy/sql/schema.py +6703 -0
  203. sqlalchemy/sql/selectable.py +7553 -0
  204. sqlalchemy/sql/sqltypes.py +4093 -0
  205. sqlalchemy/sql/traversals.py +1042 -0
  206. sqlalchemy/sql/type_api.py +2446 -0
  207. sqlalchemy/sql/util.py +1495 -0
  208. sqlalchemy/sql/visitors.py +1157 -0
  209. sqlalchemy/testing/__init__.py +96 -0
  210. sqlalchemy/testing/assertions.py +1007 -0
  211. sqlalchemy/testing/assertsql.py +519 -0
  212. sqlalchemy/testing/asyncio.py +128 -0
  213. sqlalchemy/testing/config.py +440 -0
  214. sqlalchemy/testing/engines.py +483 -0
  215. sqlalchemy/testing/entities.py +117 -0
  216. sqlalchemy/testing/exclusions.py +476 -0
  217. sqlalchemy/testing/fixtures/__init__.py +30 -0
  218. sqlalchemy/testing/fixtures/base.py +384 -0
  219. sqlalchemy/testing/fixtures/mypy.py +247 -0
  220. sqlalchemy/testing/fixtures/orm.py +227 -0
  221. sqlalchemy/testing/fixtures/sql.py +538 -0
  222. sqlalchemy/testing/pickleable.py +155 -0
  223. sqlalchemy/testing/plugin/__init__.py +6 -0
  224. sqlalchemy/testing/plugin/bootstrap.py +51 -0
  225. sqlalchemy/testing/plugin/plugin_base.py +828 -0
  226. sqlalchemy/testing/plugin/pytestplugin.py +892 -0
  227. sqlalchemy/testing/profiling.py +329 -0
  228. sqlalchemy/testing/provision.py +613 -0
  229. sqlalchemy/testing/requirements.py +1978 -0
  230. sqlalchemy/testing/schema.py +198 -0
  231. sqlalchemy/testing/suite/__init__.py +19 -0
  232. sqlalchemy/testing/suite/test_cte.py +237 -0
  233. sqlalchemy/testing/suite/test_ddl.py +420 -0
  234. sqlalchemy/testing/suite/test_dialect.py +776 -0
  235. sqlalchemy/testing/suite/test_insert.py +630 -0
  236. sqlalchemy/testing/suite/test_reflection.py +3557 -0
  237. sqlalchemy/testing/suite/test_results.py +660 -0
  238. sqlalchemy/testing/suite/test_rowcount.py +258 -0
  239. sqlalchemy/testing/suite/test_select.py +2112 -0
  240. sqlalchemy/testing/suite/test_sequence.py +317 -0
  241. sqlalchemy/testing/suite/test_table_via_select.py +686 -0
  242. sqlalchemy/testing/suite/test_types.py +2271 -0
  243. sqlalchemy/testing/suite/test_unicode_ddl.py +189 -0
  244. sqlalchemy/testing/suite/test_update_delete.py +139 -0
  245. sqlalchemy/testing/util.py +535 -0
  246. sqlalchemy/testing/warnings.py +52 -0
  247. sqlalchemy/types.py +76 -0
  248. sqlalchemy/util/__init__.py +158 -0
  249. sqlalchemy/util/_collections.py +688 -0
  250. sqlalchemy/util/_collections_cy.cp313t-win_arm64.pyd +0 -0
  251. sqlalchemy/util/_collections_cy.pxd +8 -0
  252. sqlalchemy/util/_collections_cy.py +516 -0
  253. sqlalchemy/util/_has_cython.py +46 -0
  254. sqlalchemy/util/_immutabledict_cy.cp313t-win_arm64.pyd +0 -0
  255. sqlalchemy/util/_immutabledict_cy.py +240 -0
  256. sqlalchemy/util/compat.py +299 -0
  257. sqlalchemy/util/concurrency.py +322 -0
  258. sqlalchemy/util/cython.py +79 -0
  259. sqlalchemy/util/deprecations.py +401 -0
  260. sqlalchemy/util/langhelpers.py +2320 -0
  261. sqlalchemy/util/preloaded.py +152 -0
  262. sqlalchemy/util/queue.py +304 -0
  263. sqlalchemy/util/tool_support.py +201 -0
  264. sqlalchemy/util/topological.py +120 -0
  265. sqlalchemy/util/typing.py +711 -0
  266. sqlalchemy-2.1.0b2.dist-info/METADATA +269 -0
  267. sqlalchemy-2.1.0b2.dist-info/RECORD +270 -0
  268. sqlalchemy-2.1.0b2.dist-info/WHEEL +5 -0
  269. sqlalchemy-2.1.0b2.dist-info/licenses/LICENSE +19 -0
  270. sqlalchemy-2.1.0b2.dist-info/top_level.txt +1 -0
sqlalchemy/sql/util.py ADDED
@@ -0,0 +1,1495 @@
1
+ # sql/util.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: allow-untyped-defs, allow-untyped-calls
8
+
9
+ """High level utilities which build upon other modules here."""
10
+ from __future__ import annotations
11
+
12
+ from collections import deque
13
+ import copy
14
+ from itertools import chain
15
+ import typing
16
+ from typing import AbstractSet
17
+ from typing import Any
18
+ from typing import Callable
19
+ from typing import cast
20
+ from typing import Collection
21
+ from typing import Dict
22
+ from typing import Iterable
23
+ from typing import Iterator
24
+ from typing import List
25
+ from typing import Literal
26
+ from typing import Optional
27
+ from typing import overload
28
+ from typing import Protocol
29
+ from typing import Sequence
30
+ from typing import Tuple
31
+ from typing import TYPE_CHECKING
32
+ from typing import TypeVar
33
+ from typing import Union
34
+
35
+ from . import coercions
36
+ from . import operators
37
+ from . import roles
38
+ from . import visitors
39
+ from ._typing import is_text_clause
40
+ from .annotation import _deep_annotate as _deep_annotate # noqa: F401
41
+ from .annotation import _deep_deannotate as _deep_deannotate # noqa: F401
42
+ from .annotation import _shallow_annotate as _shallow_annotate # noqa: F401
43
+ from .base import _expand_cloned
44
+ from .base import _from_objects
45
+ from .cache_key import HasCacheKey as HasCacheKey # noqa: F401
46
+ from .ddl import sort_tables as sort_tables # noqa: F401
47
+ from .elements import _find_columns as _find_columns
48
+ from .elements import _label_reference
49
+ from .elements import _textual_label_reference
50
+ from .elements import BindParameter
51
+ from .elements import ClauseElement
52
+ from .elements import ColumnClause
53
+ from .elements import ColumnElement
54
+ from .elements import Grouping
55
+ from .elements import KeyedColumnElement
56
+ from .elements import Label
57
+ from .elements import NamedColumn
58
+ from .elements import Null
59
+ from .elements import UnaryExpression
60
+ from .schema import Column
61
+ from .selectable import Alias
62
+ from .selectable import FromClause
63
+ from .selectable import FromGrouping
64
+ from .selectable import Join
65
+ from .selectable import ScalarSelect
66
+ from .selectable import SelectBase
67
+ from .selectable import TableClause
68
+ from .visitors import _ET
69
+ from .. import exc
70
+ from .. import util
71
+ from ..util.typing import Unpack
72
+
73
+ if typing.TYPE_CHECKING:
74
+ from ._typing import _EquivalentColumnMap
75
+ from ._typing import _LimitOffsetType
76
+ from ._typing import _TypeEngineArgument
77
+ from .elements import AbstractTextClause
78
+ from .elements import BinaryExpression
79
+ from .selectable import _JoinTargetElement
80
+ from .selectable import _SelectIterable
81
+ from .selectable import Selectable
82
+ from .visitors import _TraverseCallableType
83
+ from .visitors import ExternallyTraversible
84
+ from .visitors import ExternalTraversal
85
+ from ..engine.interfaces import _AnyExecuteParams
86
+ from ..engine.interfaces import _AnyMultiExecuteParams
87
+ from ..engine.interfaces import _AnySingleExecuteParams
88
+ from ..engine.interfaces import _CoreSingleExecuteParams
89
+ from ..engine.row import Row
90
+
91
+ _CE = TypeVar("_CE", bound="ColumnElement[Any]")
92
+
93
+
94
+ def join_condition(
95
+ a: FromClause,
96
+ b: FromClause,
97
+ a_subset: Optional[FromClause] = None,
98
+ consider_as_foreign_keys: Optional[AbstractSet[ColumnClause[Any]]] = None,
99
+ ) -> ColumnElement[bool]:
100
+ """Create a join condition between two tables or selectables.
101
+
102
+ e.g.::
103
+
104
+ join_condition(tablea, tableb)
105
+
106
+ would produce an expression along the lines of::
107
+
108
+ tablea.c.id == tableb.c.tablea_id
109
+
110
+ The join is determined based on the foreign key relationships
111
+ between the two selectables. If there are multiple ways
112
+ to join, or no way to join, an error is raised.
113
+
114
+ :param a_subset: An optional expression that is a sub-component
115
+ of ``a``. An attempt will be made to join to just this sub-component
116
+ first before looking at the full ``a`` construct, and if found
117
+ will be successful even if there are other ways to join to ``a``.
118
+ This allows the "right side" of a join to be passed thereby
119
+ providing a "natural join".
120
+
121
+ """
122
+ return Join._join_condition(
123
+ a,
124
+ b,
125
+ a_subset=a_subset,
126
+ consider_as_foreign_keys=consider_as_foreign_keys,
127
+ )
128
+
129
+
130
+ def find_join_source(
131
+ clauses: List[FromClause], join_to: FromClause
132
+ ) -> List[int]:
133
+ """Given a list of FROM clauses and a selectable,
134
+ return the first index and element from the list of
135
+ clauses which can be joined against the selectable. returns
136
+ None, None if no match is found.
137
+
138
+ e.g.::
139
+
140
+ clause1 = table1.join(table2)
141
+ clause2 = table4.join(table5)
142
+
143
+ join_to = table2.join(table3)
144
+
145
+ find_join_source([clause1, clause2], join_to) == clause1
146
+
147
+ """
148
+
149
+ selectables = list(_from_objects(join_to))
150
+ idx = []
151
+ for i, f in enumerate(clauses):
152
+ for s in selectables:
153
+ if f.is_derived_from(s):
154
+ idx.append(i)
155
+ return idx
156
+
157
+
158
+ def find_left_clause_that_matches_given(
159
+ clauses: Sequence[FromClause], join_from: FromClause
160
+ ) -> List[int]:
161
+ """Given a list of FROM clauses and a selectable,
162
+ return the indexes from the list of
163
+ clauses which is derived from the selectable.
164
+
165
+ """
166
+
167
+ selectables = list(_from_objects(join_from))
168
+ liberal_idx = []
169
+ for i, f in enumerate(clauses):
170
+ for s in selectables:
171
+ # basic check, if f is derived from s.
172
+ # this can be joins containing a table, or an aliased table
173
+ # or select statement matching to a table. This check
174
+ # will match a table to a selectable that is adapted from
175
+ # that table. With Query, this suits the case where a join
176
+ # is being made to an adapted entity
177
+ if f.is_derived_from(s):
178
+ liberal_idx.append(i)
179
+ break
180
+
181
+ # in an extremely small set of use cases, a join is being made where
182
+ # there are multiple FROM clauses where our target table is represented
183
+ # in more than one, such as embedded or similar. in this case, do
184
+ # another pass where we try to get a more exact match where we aren't
185
+ # looking at adaption relationships.
186
+ if len(liberal_idx) > 1:
187
+ conservative_idx = []
188
+ for idx in liberal_idx:
189
+ f = clauses[idx]
190
+ for s in selectables:
191
+ if set(surface_selectables(f)).intersection(
192
+ surface_selectables(s)
193
+ ):
194
+ conservative_idx.append(idx)
195
+ break
196
+ if conservative_idx:
197
+ return conservative_idx
198
+
199
+ return liberal_idx
200
+
201
+
202
+ def find_left_clause_to_join_from(
203
+ clauses: Sequence[FromClause],
204
+ join_to: _JoinTargetElement,
205
+ onclause: Optional[ColumnElement[Any]],
206
+ ) -> List[int]:
207
+ """Given a list of FROM clauses, a selectable,
208
+ and optional ON clause, return a list of integer indexes from the
209
+ clauses list indicating the clauses that can be joined from.
210
+
211
+ The presence of an "onclause" indicates that at least one clause can
212
+ definitely be joined from; if the list of clauses is of length one
213
+ and the onclause is given, returns that index. If the list of clauses
214
+ is more than length one, and the onclause is given, attempts to locate
215
+ which clauses contain the same columns.
216
+
217
+ """
218
+ idx = []
219
+ selectables = set(_from_objects(join_to))
220
+
221
+ # if we are given more than one target clause to join
222
+ # from, use the onclause to provide a more specific answer.
223
+ # otherwise, don't try to limit, after all, "ON TRUE" is a valid
224
+ # on clause
225
+ if len(clauses) > 1 and onclause is not None:
226
+ resolve_ambiguity = True
227
+ cols_in_onclause = _find_columns(onclause)
228
+ else:
229
+ resolve_ambiguity = False
230
+ cols_in_onclause = None
231
+
232
+ for i, f in enumerate(clauses):
233
+ for s in selectables.difference([f]):
234
+ if resolve_ambiguity:
235
+ assert cols_in_onclause is not None
236
+ if set(f.c).union(s.c).issuperset(cols_in_onclause):
237
+ idx.append(i)
238
+ break
239
+ elif onclause is not None or Join._can_join(f, s):
240
+ idx.append(i)
241
+ break
242
+
243
+ if len(idx) > 1:
244
+ # this is the same "hide froms" logic from
245
+ # Selectable._get_display_froms
246
+ toremove = set(
247
+ chain(*[_expand_cloned(f._hide_froms) for f in clauses])
248
+ )
249
+ idx = [i for i in idx if clauses[i] not in toremove]
250
+
251
+ # onclause was given and none of them resolved, so assume
252
+ # all indexes can match
253
+ if not idx and onclause is not None:
254
+ return list(range(len(clauses)))
255
+ else:
256
+ return idx
257
+
258
+
259
+ def visit_binary_product(
260
+ fn: Callable[
261
+ [BinaryExpression[Any], ColumnElement[Any], ColumnElement[Any]], None
262
+ ],
263
+ expr: ColumnElement[Any],
264
+ ) -> None:
265
+ """Produce a traversal of the given expression, delivering
266
+ column comparisons to the given function.
267
+
268
+ The function is of the form::
269
+
270
+ def my_fn(binary, left, right): ...
271
+
272
+ For each binary expression located which has a
273
+ comparison operator, the product of "left" and
274
+ "right" will be delivered to that function,
275
+ in terms of that binary.
276
+
277
+ Hence an expression like::
278
+
279
+ and_((a + b) == q + func.sum(e + f), j == r)
280
+
281
+ would have the traversal:
282
+
283
+ .. sourcecode:: text
284
+
285
+ a <eq> q
286
+ a <eq> e
287
+ a <eq> f
288
+ b <eq> q
289
+ b <eq> e
290
+ b <eq> f
291
+ j <eq> r
292
+
293
+ That is, every combination of "left" and
294
+ "right" that doesn't further contain
295
+ a binary comparison is passed as pairs.
296
+
297
+ """
298
+ stack: List[BinaryExpression[Any]] = []
299
+
300
+ def visit(element: ClauseElement) -> Iterator[ColumnElement[Any]]:
301
+ if isinstance(element, ScalarSelect):
302
+ # we don't want to dig into correlated subqueries,
303
+ # those are just column elements by themselves
304
+ yield element
305
+ elif element.__visit_name__ == "binary" and operators.is_comparison(
306
+ element.operator # type: ignore
307
+ ):
308
+ stack.insert(0, element) # type: ignore
309
+ for l in visit(element.left): # type: ignore
310
+ for r in visit(element.right): # type: ignore
311
+ fn(stack[0], l, r)
312
+ stack.pop(0)
313
+ for elem in element.get_children():
314
+ visit(elem)
315
+ else:
316
+ if isinstance(element, ColumnClause):
317
+ yield element
318
+ for elem in element.get_children():
319
+ yield from visit(elem)
320
+
321
+ list(visit(expr))
322
+ visit = None # type: ignore # remove gc cycles
323
+
324
+
325
+ def find_tables(
326
+ clause: ClauseElement,
327
+ *,
328
+ check_columns: bool = False,
329
+ include_aliases: bool = False,
330
+ include_joins: bool = False,
331
+ include_selects: bool = False,
332
+ include_crud: bool = False,
333
+ ) -> List[TableClause]:
334
+ """locate Table objects within the given expression."""
335
+
336
+ tables: List[TableClause] = []
337
+ _visitors: Dict[str, _TraverseCallableType[Any]] = {}
338
+
339
+ if include_selects:
340
+ _visitors["select"] = _visitors["compound_select"] = tables.append
341
+
342
+ if include_joins:
343
+ _visitors["join"] = tables.append
344
+
345
+ if include_aliases:
346
+ _visitors["alias"] = _visitors["subquery"] = _visitors[
347
+ "tablesample"
348
+ ] = _visitors["lateral"] = tables.append
349
+
350
+ if include_crud:
351
+ _visitors["insert"] = _visitors["update"] = _visitors["delete"] = (
352
+ lambda ent: tables.append(ent.table)
353
+ )
354
+
355
+ if check_columns:
356
+
357
+ def visit_column(column):
358
+ tables.append(column.table)
359
+
360
+ _visitors["column"] = visit_column
361
+
362
+ _visitors["table"] = tables.append
363
+
364
+ visitors.traverse(clause, {}, _visitors)
365
+ return tables
366
+
367
+
368
+ def unwrap_order_by(clause: Any) -> Any:
369
+ """Break up an 'order by' expression into individual column-expressions,
370
+ without DESC/ASC/NULLS FIRST/NULLS LAST"""
371
+
372
+ cols = util.column_set()
373
+ result = []
374
+ stack = deque([clause])
375
+
376
+ # examples
377
+ # column -> ASC/DESC == column
378
+ # column -> ASC/DESC -> label == column
379
+ # column -> label -> ASC/DESC -> label == column
380
+ # scalar_select -> label -> ASC/DESC == scalar_select -> label
381
+
382
+ while stack:
383
+ t = stack.popleft()
384
+ if isinstance(t, ColumnElement) and (
385
+ not isinstance(t, UnaryExpression)
386
+ or not operators.is_ordering_modifier(t.modifier) # type: ignore
387
+ ):
388
+ if isinstance(t, Label) and not isinstance(
389
+ t.element, ScalarSelect
390
+ ):
391
+ t = t.element
392
+
393
+ if isinstance(t, Grouping):
394
+ t = t.element
395
+
396
+ stack.append(t)
397
+ continue
398
+ elif isinstance(t, _label_reference):
399
+ t = t.element
400
+
401
+ stack.append(t)
402
+ continue
403
+ if isinstance(t, (_textual_label_reference)):
404
+ continue
405
+ if t not in cols:
406
+ cols.add(t)
407
+ result.append(t)
408
+
409
+ else:
410
+ for c in t.get_children():
411
+ stack.append(c)
412
+ return result
413
+
414
+
415
+ def unwrap_label_reference(element):
416
+ def replace(
417
+ element: ExternallyTraversible, **kw: Any
418
+ ) -> Optional[ExternallyTraversible]:
419
+ if isinstance(element, _label_reference):
420
+ return element.element
421
+ elif isinstance(element, _textual_label_reference):
422
+ assert False, "can't unwrap a textual label reference"
423
+ return None
424
+
425
+ return visitors.replacement_traverse(element, {}, replace)
426
+
427
+
428
+ def expand_column_list_from_order_by(collist, order_by):
429
+ """Given the columns clause and ORDER BY of a selectable,
430
+ return a list of column expressions that can be added to the collist
431
+ corresponding to the ORDER BY, without repeating those already
432
+ in the collist.
433
+
434
+ """
435
+ cols_already_present = {
436
+ col.element if col._order_by_label_element is not None else col
437
+ for col in collist
438
+ }
439
+
440
+ to_look_for = list(chain(*[unwrap_order_by(o) for o in order_by]))
441
+
442
+ return [col for col in to_look_for if col not in cols_already_present]
443
+
444
+
445
+ def clause_is_present(clause, search):
446
+ """Given a target clause and a second to search within, return True
447
+ if the target is plainly present in the search without any
448
+ subqueries or aliases involved.
449
+
450
+ Basically descends through Joins.
451
+
452
+ """
453
+
454
+ for elem in surface_selectables(search):
455
+ if clause == elem: # use == here so that Annotated's compare
456
+ return True
457
+ else:
458
+ return False
459
+
460
+
461
+ def tables_from_leftmost(clause: FromClause) -> Iterator[FromClause]:
462
+ if isinstance(clause, Join):
463
+ yield from tables_from_leftmost(clause.left)
464
+ yield from tables_from_leftmost(clause.right)
465
+ elif isinstance(clause, FromGrouping):
466
+ yield from tables_from_leftmost(clause.element)
467
+ else:
468
+ yield clause
469
+
470
+
471
+ def surface_expressions(clause):
472
+ stack = [clause]
473
+ while stack:
474
+ elem = stack.pop()
475
+ yield elem
476
+ if isinstance(elem, ColumnElement):
477
+ stack.extend(elem.get_children())
478
+
479
+
480
+ def surface_selectables(clause):
481
+ stack = [clause]
482
+ while stack:
483
+ elem = stack.pop()
484
+ yield elem
485
+ if isinstance(elem, Join):
486
+ stack.extend((elem.left, elem.right))
487
+ elif isinstance(elem, FromGrouping):
488
+ stack.append(elem.element)
489
+
490
+
491
+ def surface_selectables_only(clause: ClauseElement) -> Iterator[ClauseElement]:
492
+ stack = [clause]
493
+ while stack:
494
+ elem = stack.pop()
495
+ if isinstance(elem, (TableClause, Alias)):
496
+ yield elem
497
+ if isinstance(elem, Join):
498
+ stack.extend((elem.left, elem.right))
499
+ elif isinstance(elem, FromGrouping):
500
+ stack.append(elem.element)
501
+ elif isinstance(elem, ColumnClause):
502
+ if elem.table is not None:
503
+ stack.append(elem.table)
504
+ else:
505
+ yield elem
506
+ elif elem is not None:
507
+ yield elem
508
+
509
+
510
+ def extract_first_column_annotation(column, annotation_name):
511
+ filter_ = (FromGrouping, SelectBase)
512
+
513
+ stack = deque([column])
514
+ while stack:
515
+ elem = stack.popleft()
516
+ if annotation_name in elem._annotations:
517
+ return elem._annotations[annotation_name]
518
+ for sub in elem.get_children():
519
+ if isinstance(sub, filter_):
520
+ continue
521
+ stack.append(sub)
522
+ return None
523
+
524
+
525
+ def selectables_overlap(left: FromClause, right: FromClause) -> bool:
526
+ """Return True if left/right have some overlapping selectable"""
527
+
528
+ return bool(
529
+ set(surface_selectables(left)).intersection(surface_selectables(right))
530
+ )
531
+
532
+
533
+ def bind_values(clause):
534
+ """Return an ordered list of "bound" values in the given clause.
535
+
536
+ E.g.::
537
+
538
+ >>> expr = and_(table.c.foo == 5, table.c.foo == 7)
539
+ >>> bind_values(expr)
540
+ [5, 7]
541
+ """
542
+
543
+ v = []
544
+
545
+ def visit_bindparam(bind):
546
+ v.append(bind.effective_value)
547
+
548
+ visitors.traverse(clause, {}, {"bindparam": visit_bindparam})
549
+ return v
550
+
551
+
552
+ def _quote_ddl_expr(element):
553
+ if isinstance(element, str):
554
+ element = element.replace("'", "''")
555
+ return "'%s'" % element
556
+ else:
557
+ return repr(element)
558
+
559
+
560
+ class _repr_base:
561
+ _LIST: int = 0
562
+ _TUPLE: int = 1
563
+ _DICT: int = 2
564
+
565
+ __slots__ = ("max_chars",)
566
+
567
+ max_chars: int
568
+
569
+ def trunc(self, value: Any) -> str:
570
+ rep = repr(value)
571
+ lenrep = len(rep)
572
+ if lenrep > self.max_chars:
573
+ segment_length = self.max_chars // 2
574
+ rep = (
575
+ rep[0:segment_length]
576
+ + (
577
+ " ... (%d characters truncated) ... "
578
+ % (lenrep - self.max_chars)
579
+ )
580
+ + rep[-segment_length:]
581
+ )
582
+ return rep
583
+
584
+
585
+ def _repr_single_value(value):
586
+ rp = _repr_base()
587
+ rp.max_chars = 300
588
+ return rp.trunc(value)
589
+
590
+
591
+ class _repr_row(_repr_base):
592
+ """Provide a string view of a row."""
593
+
594
+ __slots__ = ("row",)
595
+
596
+ def __init__(
597
+ self, row: Row[Unpack[Tuple[Any, ...]]], max_chars: int = 300
598
+ ):
599
+ self.row = row
600
+ self.max_chars = max_chars
601
+
602
+ def __repr__(self) -> str:
603
+ trunc = self.trunc
604
+ return "(%s%s)" % (
605
+ ", ".join(trunc(value) for value in self.row),
606
+ "," if len(self.row) == 1 else "",
607
+ )
608
+
609
+
610
+ class _long_statement(str):
611
+ def __str__(self) -> str:
612
+ lself = len(self)
613
+ if lself > 500:
614
+ lleft = 250
615
+ lright = 100
616
+ trunc = lself - lleft - lright
617
+ return (
618
+ f"{self[0:lleft]} ... {trunc} "
619
+ f"characters truncated ... {self[-lright:]}"
620
+ )
621
+ else:
622
+ return str.__str__(self)
623
+
624
+
625
+ class _repr_params(_repr_base):
626
+ """Provide a string view of bound parameters.
627
+
628
+ Truncates display to a given number of 'multi' parameter sets,
629
+ as well as long values to a given number of characters.
630
+
631
+ """
632
+
633
+ __slots__ = "params", "batches", "ismulti", "max_params"
634
+
635
+ def __init__(
636
+ self,
637
+ params: Optional[_AnyExecuteParams],
638
+ batches: int,
639
+ max_params: int = 100,
640
+ max_chars: int = 300,
641
+ ismulti: Optional[bool] = None,
642
+ ):
643
+ self.params = params
644
+ self.ismulti = ismulti
645
+ self.batches = batches
646
+ self.max_chars = max_chars
647
+ self.max_params = max_params
648
+
649
+ def __repr__(self) -> str:
650
+ if self.ismulti is None:
651
+ return self.trunc(self.params)
652
+
653
+ if isinstance(self.params, list):
654
+ typ = self._LIST
655
+
656
+ elif isinstance(self.params, tuple):
657
+ typ = self._TUPLE
658
+ elif isinstance(self.params, dict):
659
+ typ = self._DICT
660
+ else:
661
+ return self.trunc(self.params)
662
+
663
+ if self.ismulti:
664
+ multi_params = cast(
665
+ "_AnyMultiExecuteParams",
666
+ self.params,
667
+ )
668
+
669
+ if len(self.params) > self.batches:
670
+ msg = (
671
+ " ... displaying %i of %i total bound parameter sets ... "
672
+ )
673
+ return " ".join(
674
+ (
675
+ self._repr_multi(
676
+ multi_params[: self.batches - 2],
677
+ typ,
678
+ )[0:-1],
679
+ msg % (self.batches, len(self.params)),
680
+ self._repr_multi(multi_params[-2:], typ)[1:],
681
+ )
682
+ )
683
+ else:
684
+ return self._repr_multi(multi_params, typ)
685
+ else:
686
+ return self._repr_params(
687
+ cast(
688
+ "_AnySingleExecuteParams",
689
+ self.params,
690
+ ),
691
+ typ,
692
+ )
693
+
694
+ def _repr_multi(
695
+ self,
696
+ multi_params: _AnyMultiExecuteParams,
697
+ typ: int,
698
+ ) -> str:
699
+ if multi_params:
700
+ if isinstance(multi_params[0], list):
701
+ elem_type = self._LIST
702
+ elif isinstance(multi_params[0], tuple):
703
+ elem_type = self._TUPLE
704
+ elif isinstance(multi_params[0], dict):
705
+ elem_type = self._DICT
706
+ else:
707
+ assert False, "Unknown parameter type %s" % (
708
+ type(multi_params[0])
709
+ )
710
+
711
+ elements = ", ".join(
712
+ self._repr_params(params, elem_type) for params in multi_params
713
+ )
714
+ else:
715
+ elements = ""
716
+
717
+ if typ == self._LIST:
718
+ return "[%s]" % elements
719
+ else:
720
+ return "(%s)" % elements
721
+
722
+ def _get_batches(self, params: Iterable[Any]) -> Any:
723
+ lparams = list(params)
724
+ lenparams = len(lparams)
725
+ if lenparams > self.max_params:
726
+ lleft = self.max_params // 2
727
+ return (
728
+ lparams[0:lleft],
729
+ lparams[-lleft:],
730
+ lenparams - self.max_params,
731
+ )
732
+ else:
733
+ return lparams, None, None
734
+
735
+ def _repr_params(
736
+ self,
737
+ params: _AnySingleExecuteParams,
738
+ typ: int,
739
+ ) -> str:
740
+ if typ is self._DICT:
741
+ return self._repr_param_dict(
742
+ cast("_CoreSingleExecuteParams", params)
743
+ )
744
+ elif typ is self._TUPLE:
745
+ return self._repr_param_tuple(cast("Sequence[Any]", params))
746
+ else:
747
+ return self._repr_param_list(params)
748
+
749
+ def _repr_param_dict(self, params: _CoreSingleExecuteParams) -> str:
750
+ trunc = self.trunc
751
+ (
752
+ items_first_batch,
753
+ items_second_batch,
754
+ trunclen,
755
+ ) = self._get_batches(params.items())
756
+
757
+ if items_second_batch:
758
+ text = "{%s" % (
759
+ ", ".join(
760
+ f"{key!r}: {trunc(value)}"
761
+ for key, value in items_first_batch
762
+ )
763
+ )
764
+ text += f" ... {trunclen} parameters truncated ... "
765
+ text += "%s}" % (
766
+ ", ".join(
767
+ f"{key!r}: {trunc(value)}"
768
+ for key, value in items_second_batch
769
+ )
770
+ )
771
+ else:
772
+ text = "{%s}" % (
773
+ ", ".join(
774
+ f"{key!r}: {trunc(value)}"
775
+ for key, value in items_first_batch
776
+ )
777
+ )
778
+ return text
779
+
780
+ def _repr_param_tuple(self, params: Sequence[Any]) -> str:
781
+ trunc = self.trunc
782
+
783
+ (
784
+ items_first_batch,
785
+ items_second_batch,
786
+ trunclen,
787
+ ) = self._get_batches(params)
788
+
789
+ if items_second_batch:
790
+ text = "(%s" % (
791
+ ", ".join(trunc(value) for value in items_first_batch)
792
+ )
793
+ text += f" ... {trunclen} parameters truncated ... "
794
+ text += "%s)" % (
795
+ ", ".join(trunc(value) for value in items_second_batch),
796
+ )
797
+ else:
798
+ text = "(%s%s)" % (
799
+ ", ".join(trunc(value) for value in items_first_batch),
800
+ "," if len(items_first_batch) == 1 else "",
801
+ )
802
+ return text
803
+
804
+ def _repr_param_list(self, params: _AnySingleExecuteParams) -> str:
805
+ trunc = self.trunc
806
+ (
807
+ items_first_batch,
808
+ items_second_batch,
809
+ trunclen,
810
+ ) = self._get_batches(params)
811
+
812
+ if items_second_batch:
813
+ text = "[%s" % (
814
+ ", ".join(trunc(value) for value in items_first_batch)
815
+ )
816
+ text += f" ... {trunclen} parameters truncated ... "
817
+ text += "%s]" % (
818
+ ", ".join(trunc(value) for value in items_second_batch)
819
+ )
820
+ else:
821
+ text = "[%s]" % (
822
+ ", ".join(trunc(value) for value in items_first_batch)
823
+ )
824
+ return text
825
+
826
+
827
+ def adapt_criterion_to_null(crit: _CE, nulls: Collection[Any]) -> _CE:
828
+ """given criterion containing bind params, convert selected elements
829
+ to IS NULL.
830
+
831
+ """
832
+
833
+ def visit_binary(binary):
834
+ if (
835
+ isinstance(binary.left, BindParameter)
836
+ and binary.left._identifying_key in nulls
837
+ ):
838
+ # reverse order if the NULL is on the left side
839
+ binary.left = binary.right
840
+ binary.right = Null()
841
+ binary.operator = operators.is_
842
+ binary.negate = operators.is_not
843
+ elif (
844
+ isinstance(binary.right, BindParameter)
845
+ and binary.right._identifying_key in nulls
846
+ ):
847
+ binary.right = Null()
848
+ binary.operator = operators.is_
849
+ binary.negate = operators.is_not
850
+
851
+ return visitors.cloned_traverse(crit, {}, {"binary": visit_binary})
852
+
853
+
854
+ def splice_joins(
855
+ left: Optional[FromClause],
856
+ right: Optional[FromClause],
857
+ stop_on: Optional[FromClause] = None,
858
+ ) -> Optional[FromClause]:
859
+ if left is None:
860
+ return right
861
+
862
+ stack: List[Tuple[Optional[FromClause], Optional[Join]]] = [(right, None)]
863
+
864
+ adapter = ClauseAdapter(left)
865
+ ret = None
866
+ while stack:
867
+ (right, prevright) = stack.pop()
868
+ if isinstance(right, Join) and right is not stop_on:
869
+ right = right._clone()
870
+ right.onclause = adapter.traverse(right.onclause)
871
+ stack.append((right.left, right))
872
+ else:
873
+ right = adapter.traverse(right)
874
+ if prevright is not None:
875
+ assert right is not None
876
+ prevright.left = right
877
+ if ret is None:
878
+ ret = right
879
+
880
+ return ret
881
+
882
+
883
+ @overload
884
+ def reduce_columns(
885
+ columns: Iterable[ColumnElement[Any]],
886
+ *clauses: Optional[ClauseElement],
887
+ **kw: bool,
888
+ ) -> Sequence[ColumnElement[Any]]: ...
889
+
890
+
891
+ @overload
892
+ def reduce_columns(
893
+ columns: _SelectIterable,
894
+ *clauses: Optional[ClauseElement],
895
+ **kw: bool,
896
+ ) -> Sequence[Union[ColumnElement[Any], AbstractTextClause]]: ...
897
+
898
+
899
+ def reduce_columns(
900
+ columns: _SelectIterable,
901
+ *clauses: Optional[ClauseElement],
902
+ **kw: bool,
903
+ ) -> Collection[Union[ColumnElement[Any], AbstractTextClause]]:
904
+ r"""given a list of columns, return a 'reduced' set based on natural
905
+ equivalents.
906
+
907
+ the set is reduced to the smallest list of columns which have no natural
908
+ equivalent present in the list. A "natural equivalent" means that two
909
+ columns will ultimately represent the same value because they are related
910
+ by a foreign key.
911
+
912
+ \*clauses is an optional list of join clauses which will be traversed
913
+ to further identify columns that are "equivalent".
914
+
915
+ \**kw may specify 'ignore_nonexistent_tables' to ignore foreign keys
916
+ whose tables are not yet configured, or columns that aren't yet present.
917
+
918
+ This function is primarily used to determine the most minimal "primary
919
+ key" from a selectable, by reducing the set of primary key columns present
920
+ in the selectable to just those that are not repeated.
921
+
922
+ """
923
+ ignore_nonexistent_tables = kw.pop("ignore_nonexistent_tables", False)
924
+ only_synonyms = kw.pop("only_synonyms", False)
925
+
926
+ column_set = util.OrderedSet(columns)
927
+ cset_no_text: util.OrderedSet[ColumnElement[Any]] = column_set.difference(
928
+ c for c in column_set if is_text_clause(c) # type: ignore
929
+ )
930
+
931
+ omit = util.column_set()
932
+ for col in cset_no_text:
933
+ for fk in chain(*[c.foreign_keys for c in col.proxy_set]):
934
+ for c in cset_no_text:
935
+ if c is col:
936
+ continue
937
+ try:
938
+ fk_col = fk.column
939
+ except exc.NoReferencedColumnError:
940
+ # TODO: add specific coverage here
941
+ # to test/sql/test_selectable ReduceTest
942
+ if ignore_nonexistent_tables:
943
+ continue
944
+ else:
945
+ raise
946
+ except exc.NoReferencedTableError:
947
+ # TODO: add specific coverage here
948
+ # to test/sql/test_selectable ReduceTest
949
+ if ignore_nonexistent_tables:
950
+ continue
951
+ else:
952
+ raise
953
+ if fk_col.shares_lineage(c) and (
954
+ not only_synonyms or c.name == col.name
955
+ ):
956
+ omit.add(col)
957
+ break
958
+
959
+ if clauses:
960
+
961
+ def visit_binary(binary):
962
+ if binary.operator == operators.eq:
963
+ cols = util.column_set(
964
+ chain(
965
+ *[c.proxy_set for c in cset_no_text.difference(omit)]
966
+ )
967
+ )
968
+ if binary.left in cols and binary.right in cols:
969
+ for c in reversed(cset_no_text):
970
+ if c.shares_lineage(binary.right) and (
971
+ not only_synonyms or c.name == binary.left.name
972
+ ):
973
+ omit.add(c)
974
+ break
975
+
976
+ for clause in clauses:
977
+ if clause is not None:
978
+ visitors.traverse(clause, {}, {"binary": visit_binary})
979
+
980
+ return column_set.difference(omit)
981
+
982
+
983
+ def criterion_as_pairs(
984
+ expression,
985
+ consider_as_foreign_keys=None,
986
+ consider_as_referenced_keys=None,
987
+ any_operator=False,
988
+ ):
989
+ """traverse an expression and locate binary criterion pairs."""
990
+
991
+ if consider_as_foreign_keys and consider_as_referenced_keys:
992
+ raise exc.ArgumentError(
993
+ "Can only specify one of "
994
+ "'consider_as_foreign_keys' or "
995
+ "'consider_as_referenced_keys'"
996
+ )
997
+
998
+ def col_is(a, b):
999
+ # return a is b
1000
+ return a.compare(b)
1001
+
1002
+ def visit_binary(binary):
1003
+ if not any_operator and binary.operator is not operators.eq:
1004
+ return
1005
+ if not isinstance(binary.left, ColumnElement) or not isinstance(
1006
+ binary.right, ColumnElement
1007
+ ):
1008
+ return
1009
+
1010
+ if consider_as_foreign_keys:
1011
+ if binary.left in consider_as_foreign_keys and (
1012
+ col_is(binary.right, binary.left)
1013
+ or binary.right not in consider_as_foreign_keys
1014
+ ):
1015
+ pairs.append((binary.right, binary.left))
1016
+ elif binary.right in consider_as_foreign_keys and (
1017
+ col_is(binary.left, binary.right)
1018
+ or binary.left not in consider_as_foreign_keys
1019
+ ):
1020
+ pairs.append((binary.left, binary.right))
1021
+ elif consider_as_referenced_keys:
1022
+ if binary.left in consider_as_referenced_keys and (
1023
+ col_is(binary.right, binary.left)
1024
+ or binary.right not in consider_as_referenced_keys
1025
+ ):
1026
+ pairs.append((binary.left, binary.right))
1027
+ elif binary.right in consider_as_referenced_keys and (
1028
+ col_is(binary.left, binary.right)
1029
+ or binary.left not in consider_as_referenced_keys
1030
+ ):
1031
+ pairs.append((binary.right, binary.left))
1032
+ else:
1033
+ if isinstance(binary.left, Column) and isinstance(
1034
+ binary.right, Column
1035
+ ):
1036
+ if binary.left.references(binary.right):
1037
+ pairs.append((binary.right, binary.left))
1038
+ elif binary.right.references(binary.left):
1039
+ pairs.append((binary.left, binary.right))
1040
+
1041
+ pairs: List[Tuple[ColumnElement[Any], ColumnElement[Any]]] = []
1042
+ visitors.traverse(expression, {}, {"binary": visit_binary})
1043
+ return pairs
1044
+
1045
+
1046
+ class ClauseAdapter(visitors.ReplacingExternalTraversal):
1047
+ """Clones and modifies clauses based on column correspondence.
1048
+
1049
+ E.g.::
1050
+
1051
+ table1 = Table(
1052
+ "sometable",
1053
+ metadata,
1054
+ Column("col1", Integer),
1055
+ Column("col2", Integer),
1056
+ )
1057
+ table2 = Table(
1058
+ "someothertable",
1059
+ metadata,
1060
+ Column("col1", Integer),
1061
+ Column("col2", Integer),
1062
+ )
1063
+
1064
+ condition = table1.c.col1 == table2.c.col1
1065
+
1066
+ make an alias of table1::
1067
+
1068
+ s = table1.alias("foo")
1069
+
1070
+ calling ``ClauseAdapter(s).traverse(condition)`` converts
1071
+ condition to read::
1072
+
1073
+ s.c.col1 == table2.c.col1
1074
+
1075
+ """
1076
+
1077
+ __slots__ = (
1078
+ "__traverse_options__",
1079
+ "selectable",
1080
+ "include_fn",
1081
+ "exclude_fn",
1082
+ "equivalents",
1083
+ "adapt_on_names",
1084
+ "adapt_from_selectables",
1085
+ )
1086
+
1087
+ def __init__(
1088
+ self,
1089
+ selectable: Selectable,
1090
+ equivalents: Optional[_EquivalentColumnMap] = None,
1091
+ include_fn: Optional[Callable[[ClauseElement], bool]] = None,
1092
+ exclude_fn: Optional[Callable[[ClauseElement], bool]] = None,
1093
+ adapt_on_names: bool = False,
1094
+ anonymize_labels: bool = False,
1095
+ adapt_from_selectables: Optional[AbstractSet[FromClause]] = None,
1096
+ ):
1097
+ self.__traverse_options__ = {
1098
+ "stop_on": [selectable],
1099
+ "anonymize_labels": anonymize_labels,
1100
+ }
1101
+ self.selectable = selectable
1102
+ self.include_fn = include_fn
1103
+ self.exclude_fn = exclude_fn
1104
+ self.equivalents = util.column_dict(equivalents or {})
1105
+ self.adapt_on_names = adapt_on_names
1106
+ self.adapt_from_selectables = adapt_from_selectables
1107
+
1108
+ if TYPE_CHECKING:
1109
+
1110
+ @overload
1111
+ def traverse(self, obj: Literal[None]) -> None: ...
1112
+
1113
+ # note this specializes the ReplacingExternalTraversal.traverse()
1114
+ # method to state
1115
+ # that we will return the same kind of ExternalTraversal object as
1116
+ # we were given. This is probably not 100% true, such as it's
1117
+ # possible for us to swap out Alias for Table at the top level.
1118
+ # Ideally there could be overloads specific to ColumnElement and
1119
+ # FromClause but Mypy is not accepting those as compatible with
1120
+ # the base ReplacingExternalTraversal
1121
+ @overload
1122
+ def traverse(self, obj: _ET) -> _ET: ...
1123
+
1124
+ def traverse(
1125
+ self, obj: Optional[ExternallyTraversible]
1126
+ ) -> Optional[ExternallyTraversible]: ...
1127
+
1128
+ def _corresponding_column(
1129
+ self, col, require_embedded, _seen=util.EMPTY_SET
1130
+ ):
1131
+ newcol = self.selectable.corresponding_column(
1132
+ col, require_embedded=require_embedded
1133
+ )
1134
+ if newcol is None and col in self.equivalents and col not in _seen:
1135
+ for equiv in self.equivalents[col]:
1136
+ newcol = self._corresponding_column(
1137
+ equiv,
1138
+ require_embedded=require_embedded,
1139
+ _seen=_seen.union([col]),
1140
+ )
1141
+ if newcol is not None:
1142
+ return newcol
1143
+
1144
+ if (
1145
+ self.adapt_on_names
1146
+ and newcol is None
1147
+ and isinstance(col, NamedColumn)
1148
+ ):
1149
+ newcol = self.selectable.exported_columns.get(col.name)
1150
+ return newcol
1151
+
1152
+ @util.preload_module("sqlalchemy.sql.functions")
1153
+ def replace(
1154
+ self, col: _ET, _include_singleton_constants: bool = False
1155
+ ) -> Optional[_ET]:
1156
+ functions = util.preloaded.sql_functions
1157
+
1158
+ # TODO: cython candidate
1159
+
1160
+ if self.include_fn and not self.include_fn(col): # type: ignore
1161
+ return None
1162
+ elif self.exclude_fn and self.exclude_fn(col): # type: ignore
1163
+ return None
1164
+
1165
+ if isinstance(col, FromClause) and not isinstance(
1166
+ col, functions.FunctionElement
1167
+ ):
1168
+ if self.selectable.is_derived_from(col):
1169
+ if self.adapt_from_selectables:
1170
+ for adp in self.adapt_from_selectables:
1171
+ if adp.is_derived_from(col):
1172
+ break
1173
+ else:
1174
+ return None
1175
+ return self.selectable # type: ignore
1176
+ elif isinstance(col, Alias) and isinstance(
1177
+ col.element, TableClause
1178
+ ):
1179
+ # we are a SELECT statement and not derived from an alias of a
1180
+ # table (which nonetheless may be a table our SELECT derives
1181
+ # from), so return the alias to prevent further traversal
1182
+ # or
1183
+ # we are an alias of a table and we are not derived from an
1184
+ # alias of a table (which nonetheless may be the same table
1185
+ # as ours) so, same thing
1186
+ return col
1187
+ else:
1188
+ # other cases where we are a selectable and the element
1189
+ # is another join or selectable that contains a table which our
1190
+ # selectable derives from, that we want to process
1191
+ return None
1192
+
1193
+ elif not isinstance(col, ColumnElement):
1194
+ return None
1195
+ elif not _include_singleton_constants and col._is_singleton_constant:
1196
+ # dont swap out NULL, TRUE, FALSE for a label name
1197
+ # in a SQL statement that's being rewritten,
1198
+ # leave them as the constant. This is first noted in #6259,
1199
+ # however the logic to check this moved here as of #7154 so that
1200
+ # it is made specific to SQL rewriting and not all column
1201
+ # correspondence
1202
+
1203
+ return None
1204
+
1205
+ if "adapt_column" in col._annotations:
1206
+ col = col._annotations["adapt_column"]
1207
+
1208
+ if TYPE_CHECKING:
1209
+ assert isinstance(col, KeyedColumnElement)
1210
+
1211
+ if self.adapt_from_selectables and col not in self.equivalents:
1212
+ for adp in self.adapt_from_selectables:
1213
+ if adp.c.corresponding_column(col, False) is not None:
1214
+ break
1215
+ else:
1216
+ return None
1217
+
1218
+ if TYPE_CHECKING:
1219
+ assert isinstance(col, KeyedColumnElement)
1220
+
1221
+ return self._corresponding_column( # type: ignore
1222
+ col, require_embedded=True
1223
+ )
1224
+
1225
+
1226
+ class _ColumnLookup(Protocol):
1227
+ @overload
1228
+ def __getitem__(self, key: None) -> None: ...
1229
+
1230
+ @overload
1231
+ def __getitem__(self, key: ColumnClause[Any]) -> ColumnClause[Any]: ...
1232
+
1233
+ @overload
1234
+ def __getitem__(self, key: ColumnElement[Any]) -> ColumnElement[Any]: ...
1235
+
1236
+ @overload
1237
+ def __getitem__(self, key: _ET) -> _ET: ...
1238
+
1239
+ def __getitem__(self, key: Any) -> Any: ...
1240
+
1241
+
1242
+ class ColumnAdapter(ClauseAdapter):
1243
+ """Extends ClauseAdapter with extra utility functions.
1244
+
1245
+ Key aspects of ColumnAdapter include:
1246
+
1247
+ * Expressions that are adapted are stored in a persistent
1248
+ .columns collection; so that an expression E adapted into
1249
+ an expression E1, will return the same object E1 when adapted
1250
+ a second time. This is important in particular for things like
1251
+ Label objects that are anonymized, so that the ColumnAdapter can
1252
+ be used to present a consistent "adapted" view of things.
1253
+
1254
+ * Exclusion of items from the persistent collection based on
1255
+ include/exclude rules, but also independent of hash identity.
1256
+ This because "annotated" items all have the same hash identity as their
1257
+ parent.
1258
+
1259
+ * "wrapping" capability is added, so that the replacement of an expression
1260
+ E can proceed through a series of adapters. This differs from the
1261
+ visitor's "chaining" feature in that the resulting object is passed
1262
+ through all replacing functions unconditionally, rather than stopping
1263
+ at the first one that returns non-None.
1264
+
1265
+ * An adapt_required option, used by eager loading to indicate that
1266
+ We don't trust a result row column that is not translated.
1267
+ This is to prevent a column from being interpreted as that
1268
+ of the child row in a self-referential scenario, see
1269
+ inheritance/test_basic.py->EagerTargetingTest.test_adapt_stringency
1270
+
1271
+ """
1272
+
1273
+ __slots__ = (
1274
+ "columns",
1275
+ "adapt_required",
1276
+ "allow_label_resolve",
1277
+ "_wrap",
1278
+ "__weakref__",
1279
+ )
1280
+
1281
+ columns: _ColumnLookup
1282
+
1283
+ def __init__(
1284
+ self,
1285
+ selectable: Selectable,
1286
+ equivalents: Optional[_EquivalentColumnMap] = None,
1287
+ adapt_required: bool = False,
1288
+ include_fn: Optional[Callable[[ClauseElement], bool]] = None,
1289
+ exclude_fn: Optional[Callable[[ClauseElement], bool]] = None,
1290
+ adapt_on_names: bool = False,
1291
+ allow_label_resolve: bool = True,
1292
+ anonymize_labels: bool = False,
1293
+ adapt_from_selectables: Optional[AbstractSet[FromClause]] = None,
1294
+ ):
1295
+ super().__init__(
1296
+ selectable,
1297
+ equivalents,
1298
+ include_fn=include_fn,
1299
+ exclude_fn=exclude_fn,
1300
+ adapt_on_names=adapt_on_names,
1301
+ anonymize_labels=anonymize_labels,
1302
+ adapt_from_selectables=adapt_from_selectables,
1303
+ )
1304
+
1305
+ self.columns = util.WeakPopulateDict(self._locate_col) # type: ignore
1306
+ if self.include_fn or self.exclude_fn:
1307
+ self.columns = self._IncludeExcludeMapping(self, self.columns)
1308
+ self.adapt_required = adapt_required
1309
+ self.allow_label_resolve = allow_label_resolve
1310
+ self._wrap = None
1311
+
1312
+ class _IncludeExcludeMapping:
1313
+ def __init__(self, parent, columns):
1314
+ self.parent = parent
1315
+ self.columns = columns
1316
+
1317
+ def __getitem__(self, key):
1318
+ if (
1319
+ self.parent.include_fn and not self.parent.include_fn(key)
1320
+ ) or (self.parent.exclude_fn and self.parent.exclude_fn(key)):
1321
+ if self.parent._wrap:
1322
+ return self.parent._wrap.columns[key]
1323
+ else:
1324
+ return key
1325
+ return self.columns[key]
1326
+
1327
+ def wrap(self, adapter):
1328
+ ac = copy.copy(self)
1329
+ ac._wrap = adapter
1330
+ ac.columns = util.WeakPopulateDict(ac._locate_col) # type: ignore
1331
+ if ac.include_fn or ac.exclude_fn:
1332
+ ac.columns = self._IncludeExcludeMapping(ac, ac.columns)
1333
+
1334
+ return ac
1335
+
1336
+ @overload
1337
+ def traverse(self, obj: Literal[None]) -> None: ...
1338
+
1339
+ @overload
1340
+ def traverse(self, obj: _ET) -> _ET: ...
1341
+
1342
+ def traverse(
1343
+ self, obj: Optional[ExternallyTraversible]
1344
+ ) -> Optional[ExternallyTraversible]:
1345
+ return self.columns[obj]
1346
+
1347
+ def chain(self, visitor: ExternalTraversal) -> ColumnAdapter:
1348
+ assert isinstance(visitor, ColumnAdapter)
1349
+
1350
+ return super().chain(visitor)
1351
+
1352
+ if TYPE_CHECKING:
1353
+
1354
+ @property
1355
+ def visitor_iterator(self) -> Iterator[ColumnAdapter]: ...
1356
+
1357
+ adapt_clause = traverse
1358
+ adapt_list = ClauseAdapter.copy_and_process
1359
+
1360
+ def adapt_check_present(self, col: _ET) -> Optional[_ET]:
1361
+ newcol = self.columns[col]
1362
+
1363
+ if newcol is col and self._corresponding_column(col, True) is None:
1364
+ return None
1365
+
1366
+ return newcol
1367
+
1368
+ def _locate_col(
1369
+ self, col: ColumnElement[Any]
1370
+ ) -> Optional[ColumnElement[Any]]:
1371
+ # both replace and traverse() are overly complicated for what
1372
+ # we are doing here and we would do better to have an inlined
1373
+ # version that doesn't build up as much overhead. the issue is that
1374
+ # sometimes the lookup does in fact have to adapt the insides of
1375
+ # say a labeled scalar subquery. However, if the object is an
1376
+ # Immutable, i.e. Column objects, we can skip the "clone" /
1377
+ # "copy internals" part since those will be no-ops in any case.
1378
+ # additionally we want to catch singleton objects null/true/false
1379
+ # and make sure they are adapted as well here.
1380
+
1381
+ if col._is_immutable:
1382
+ for vis in self.visitor_iterator:
1383
+ c = vis.replace(col, _include_singleton_constants=True)
1384
+ if c is not None:
1385
+ break
1386
+ else:
1387
+ c = col
1388
+ else:
1389
+ c = ClauseAdapter.traverse(self, col)
1390
+
1391
+ if self._wrap:
1392
+ c2 = self._wrap._locate_col(c)
1393
+ if c2 is not None:
1394
+ c = c2
1395
+
1396
+ if self.adapt_required and c is col:
1397
+ return None
1398
+
1399
+ # allow_label_resolve is consumed by one case for joined eager loading
1400
+ # as part of its logic to prevent its own columns from being affected
1401
+ # by .order_by(). Before full typing were applied to the ORM, this
1402
+ # logic would set this attribute on the incoming object (which is
1403
+ # typically a column, but we have a test for it being a non-column
1404
+ # object) if no column were found. While this seemed to
1405
+ # have no negative effects, this adjustment should only occur on the
1406
+ # new column which is assumed to be local to an adapted selectable.
1407
+ if c is not col:
1408
+ c._allow_label_resolve = self.allow_label_resolve
1409
+
1410
+ return c
1411
+
1412
+
1413
+ def _offset_or_limit_clause(
1414
+ element: _LimitOffsetType,
1415
+ name: Optional[str] = None,
1416
+ type_: Optional[_TypeEngineArgument[int]] = None,
1417
+ ) -> ColumnElement[int]:
1418
+ """Convert the given value to an "offset or limit" clause.
1419
+
1420
+ This handles incoming integers and converts to an expression; if
1421
+ an expression is already given, it is passed through.
1422
+
1423
+ """
1424
+ return coercions.expect(
1425
+ roles.LimitOffsetRole, element, name=name, type_=type_
1426
+ )
1427
+
1428
+
1429
+ def _offset_or_limit_clause_asint_if_possible(
1430
+ clause: _LimitOffsetType,
1431
+ ) -> _LimitOffsetType:
1432
+ """Return the offset or limit clause as a simple integer if possible,
1433
+ else return the clause.
1434
+
1435
+ """
1436
+ if clause is None:
1437
+ return None
1438
+ if hasattr(clause, "_limit_offset_value"):
1439
+ value = clause._limit_offset_value
1440
+ return util.asint(value)
1441
+ else:
1442
+ return clause
1443
+
1444
+
1445
+ def _make_slice(
1446
+ limit_clause: _LimitOffsetType,
1447
+ offset_clause: _LimitOffsetType,
1448
+ start: int,
1449
+ stop: int,
1450
+ ) -> Tuple[Optional[ColumnElement[int]], Optional[ColumnElement[int]]]:
1451
+ """Compute LIMIT/OFFSET in terms of slice start/end"""
1452
+
1453
+ # for calculated limit/offset, try to do the addition of
1454
+ # values to offset in Python, however if a SQL clause is present
1455
+ # then the addition has to be on the SQL side.
1456
+
1457
+ # TODO: typing is finding a few gaps in here, see if they can be
1458
+ # closed up
1459
+
1460
+ if start is not None and stop is not None:
1461
+ offset_clause = _offset_or_limit_clause_asint_if_possible(
1462
+ offset_clause
1463
+ )
1464
+ if offset_clause is None:
1465
+ offset_clause = 0
1466
+
1467
+ if start != 0:
1468
+ offset_clause = offset_clause + start # type: ignore
1469
+
1470
+ if offset_clause == 0:
1471
+ offset_clause = None
1472
+ else:
1473
+ assert offset_clause is not None
1474
+ offset_clause = _offset_or_limit_clause(offset_clause)
1475
+
1476
+ limit_clause = _offset_or_limit_clause(stop - start)
1477
+
1478
+ elif start is None and stop is not None:
1479
+ limit_clause = _offset_or_limit_clause(stop)
1480
+ elif start is not None and stop is None:
1481
+ offset_clause = _offset_or_limit_clause_asint_if_possible(
1482
+ offset_clause
1483
+ )
1484
+ if offset_clause is None:
1485
+ offset_clause = 0
1486
+
1487
+ if start != 0:
1488
+ offset_clause = offset_clause + start
1489
+
1490
+ if offset_clause == 0:
1491
+ offset_clause = None
1492
+ else:
1493
+ offset_clause = _offset_or_limit_clause(offset_clause)
1494
+
1495
+ return limit_clause, offset_clause