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,434 @@
1
+ # testing/config.py
2
+ # Copyright (C) 2005-2026 the SQLAlchemy authors and contributors
3
+ # <see AUTHORS file>
4
+ #
5
+ # This module is part of SQLAlchemy and is released under
6
+ # the MIT License: https://www.opensource.org/licenses/mit-license.php
7
+ # mypy: ignore-errors
8
+
9
+
10
+ from __future__ import annotations
11
+
12
+ from argparse import Namespace
13
+ import collections
14
+ import inspect
15
+ import typing
16
+ from typing import Any
17
+ from typing import Callable
18
+ from typing import Iterable
19
+ from typing import NoReturn
20
+ from typing import Optional
21
+ from typing import Tuple
22
+ from typing import TypeVar
23
+ from typing import Union
24
+
25
+ from . import mock
26
+ from . import requirements as _requirements
27
+ from .util import fail
28
+ from .. import util
29
+
30
+ # default requirements; this is replaced by plugin_base when pytest
31
+ # is run
32
+ requirements = _requirements.SuiteRequirements()
33
+
34
+ db = None
35
+ db_url = None
36
+ db_opts = None
37
+ file_config = None
38
+ test_schema = None
39
+ test_schema_2 = None
40
+ any_async = False
41
+ _current = None
42
+ ident = "main"
43
+ options: Namespace = None # type: ignore
44
+
45
+ if typing.TYPE_CHECKING:
46
+ from .plugin.plugin_base import FixtureFunctions
47
+
48
+ _fixture_functions: FixtureFunctions
49
+ else:
50
+
51
+ class _NullFixtureFunctions:
52
+ def _null_decorator(self):
53
+ def go(fn):
54
+ return fn
55
+
56
+ return go
57
+
58
+ def skip_test_exception(self, *arg, **kw):
59
+ return Exception()
60
+
61
+ @property
62
+ def add_to_marker(self):
63
+ return mock.Mock()
64
+
65
+ def mark_base_test_class(self):
66
+ return self._null_decorator()
67
+
68
+ def combinations(self, *arg_sets, **kw):
69
+ return self._null_decorator()
70
+
71
+ def param_ident(self, *parameters):
72
+ return self._null_decorator()
73
+
74
+ def fixture(self, *arg, **kw):
75
+ return self._null_decorator()
76
+
77
+ def get_current_test_name(self):
78
+ return None
79
+
80
+ def async_test(self, fn):
81
+ return fn
82
+
83
+ # default fixture functions; these are replaced by plugin_base when
84
+ # pytest runs
85
+ _fixture_functions = _NullFixtureFunctions()
86
+
87
+
88
+ _FN = TypeVar("_FN", bound=Callable[..., Any])
89
+
90
+
91
+ def combinations(
92
+ *comb: Union[Any, Tuple[Any, ...]],
93
+ argnames: Optional[str] = None,
94
+ id_: Optional[str] = None,
95
+ **kw: str,
96
+ ) -> Callable[[_FN], _FN]:
97
+ r"""Deliver multiple versions of a test based on positional combinations.
98
+
99
+ This is a facade over pytest.mark.parametrize.
100
+
101
+
102
+ :param \*comb: argument combinations. These are tuples that will be passed
103
+ positionally to the decorated function.
104
+
105
+ :param argnames: optional list of argument names. These are the names
106
+ of the arguments in the test function that correspond to the entries
107
+ in each argument tuple. pytest.mark.parametrize requires this, however
108
+ the combinations function will derive it automatically if not present
109
+ by using ``inspect.getfullargspec(fn).args[1:]``. Note this assumes the
110
+ first argument is "self" which is discarded.
111
+
112
+ :param id\_: optional id template. This is a string template that
113
+ describes how the "id" for each parameter set should be defined, if any.
114
+ The number of characters in the template should match the number of
115
+ entries in each argument tuple. Each character describes how the
116
+ corresponding entry in the argument tuple should be handled, as far as
117
+ whether or not it is included in the arguments passed to the function, as
118
+ well as if it is included in the tokens used to create the id of the
119
+ parameter set.
120
+
121
+ If omitted, the argument combinations are passed to parametrize as is. If
122
+ passed, each argument combination is turned into a pytest.param() object,
123
+ mapping the elements of the argument tuple to produce an id based on a
124
+ character value in the same position within the string template using the
125
+ following scheme:
126
+
127
+ .. sourcecode:: text
128
+
129
+ i - the given argument is a string that is part of the id only, don't
130
+ pass it as an argument
131
+
132
+ n - the given argument should be passed and it should be added to the
133
+ id by calling the .__name__ attribute
134
+
135
+ r - the given argument should be passed and it should be added to the
136
+ id by calling repr()
137
+
138
+ s - the given argument should be passed and it should be added to the
139
+ id by calling str()
140
+
141
+ a - (argument) the given argument should be passed and it should not
142
+ be used to generated the id
143
+
144
+ e.g.::
145
+
146
+ @testing.combinations(
147
+ (operator.eq, "eq"),
148
+ (operator.ne, "ne"),
149
+ (operator.gt, "gt"),
150
+ (operator.lt, "lt"),
151
+ id_="na",
152
+ )
153
+ def test_operator(self, opfunc, name):
154
+ pass
155
+
156
+ The above combination will call ``.__name__`` on the first member of
157
+ each tuple and use that as the "id" to pytest.param().
158
+
159
+
160
+ """
161
+ return _fixture_functions.combinations(
162
+ *comb, id_=id_, argnames=argnames, **kw
163
+ )
164
+
165
+
166
+ def combinations_list(arg_iterable: Iterable[Tuple[Any, ...]], **kw):
167
+ "As combination, but takes a single iterable"
168
+ return combinations(*arg_iterable, **kw)
169
+
170
+
171
+ class Variation:
172
+ __slots__ = ("_name", "_argname")
173
+
174
+ def __init__(self, case, argname, case_names):
175
+ self._name = case
176
+ self._argname = argname
177
+ for casename in case_names:
178
+ setattr(self, casename, casename == case)
179
+
180
+ if typing.TYPE_CHECKING:
181
+
182
+ def __getattr__(self, key: str) -> bool: ...
183
+
184
+ @property
185
+ def name(self):
186
+ return self._name
187
+
188
+ def __bool__(self):
189
+ return self._name == self._argname
190
+
191
+ def __nonzero__(self):
192
+ return not self.__bool__()
193
+
194
+ def __str__(self):
195
+ return f"{self._argname}={self._name!r}"
196
+
197
+ def __repr__(self):
198
+ return str(self)
199
+
200
+ def fail(self) -> NoReturn:
201
+ fail(f"Unknown {self}")
202
+
203
+ @classmethod
204
+ def idfn(cls, variation):
205
+ return variation.name
206
+
207
+ @classmethod
208
+ def generate_cases(cls, argname, cases):
209
+ case_names = [
210
+ argname if c is True else "not_" + argname if c is False else c
211
+ for c in cases
212
+ ]
213
+
214
+ typ = type(
215
+ argname,
216
+ (Variation,),
217
+ {
218
+ "__slots__": tuple(case_names),
219
+ },
220
+ )
221
+
222
+ return [typ(casename, argname, case_names) for casename in case_names]
223
+
224
+
225
+ def variation(argname_or_fn, cases=None):
226
+ """a helper around testing.combinations that provides a single namespace
227
+ that can be used as a switch.
228
+
229
+ e.g.::
230
+
231
+ @testing.variation("querytyp", ["select", "subquery", "legacy_query"])
232
+ @testing.variation("lazy", ["select", "raise", "raise_on_sql"])
233
+ def test_thing(self, querytyp, lazy, decl_base):
234
+ class Thing(decl_base):
235
+ __tablename__ = "thing"
236
+
237
+ # use name directly
238
+ rel = relationship("Rel", lazy=lazy.name)
239
+
240
+ # use as a switch
241
+ if querytyp.select:
242
+ stmt = select(Thing)
243
+ elif querytyp.subquery:
244
+ stmt = select(Thing).subquery()
245
+ elif querytyp.legacy_query:
246
+ stmt = Session.query(Thing)
247
+ else:
248
+ querytyp.fail()
249
+
250
+ The variable provided is a slots object of boolean variables, as well
251
+ as the name of the case itself under the attribute ".name"
252
+
253
+ """
254
+
255
+ if inspect.isfunction(argname_or_fn):
256
+ argname = argname_or_fn.__name__
257
+ cases = argname_or_fn(None)
258
+
259
+ @variation_fixture(argname, cases)
260
+ def go(self, request):
261
+ yield request.param
262
+
263
+ return go
264
+ else:
265
+ argname = argname_or_fn
266
+ cases_plus_limitations = [
267
+ (
268
+ entry
269
+ if (isinstance(entry, tuple) and len(entry) == 2)
270
+ else (entry, None)
271
+ )
272
+ for entry in cases
273
+ ]
274
+
275
+ variations = Variation.generate_cases(
276
+ argname, [c for c, l in cases_plus_limitations]
277
+ )
278
+ return combinations(
279
+ *[
280
+ (
281
+ (variation._name, variation, limitation)
282
+ if limitation is not None
283
+ else (variation._name, variation)
284
+ )
285
+ for variation, (case, limitation) in zip(
286
+ variations, cases_plus_limitations
287
+ )
288
+ ],
289
+ id_="ia",
290
+ argnames=argname,
291
+ )
292
+
293
+
294
+ def variation_fixture(argname, cases, scope="function"):
295
+ return fixture(
296
+ params=Variation.generate_cases(argname, cases),
297
+ ids=Variation.idfn,
298
+ scope=scope,
299
+ )
300
+
301
+
302
+ def fixture(*arg: Any, **kw: Any) -> Any:
303
+ return _fixture_functions.fixture(*arg, **kw)
304
+
305
+
306
+ def get_current_test_name() -> str:
307
+ return _fixture_functions.get_current_test_name()
308
+
309
+
310
+ def mark_base_test_class() -> Any:
311
+ return _fixture_functions.mark_base_test_class()
312
+
313
+
314
+ class _AddToMarker:
315
+ def __getattr__(self, attr: str) -> Any:
316
+ return getattr(_fixture_functions.add_to_marker, attr)
317
+
318
+
319
+ add_to_marker = _AddToMarker()
320
+
321
+
322
+ class Config:
323
+ def __init__(self, db, db_opts, options, file_config):
324
+ self._set_name(db)
325
+ self.db = db
326
+ self.db_opts = db_opts
327
+ self.options = options
328
+ self.file_config = file_config
329
+ self.test_schema = "test_schema"
330
+ self.test_schema_2 = "test_schema_2"
331
+
332
+ self.is_async = db.dialect.is_async and not util.asbool(
333
+ db.url.query.get("async_fallback", False)
334
+ )
335
+
336
+ from . import provision
337
+
338
+ self.is_default_dialect = provision.is_preferred_driver(self, db)
339
+
340
+ _stack = collections.deque()
341
+ _configs = set()
342
+
343
+ def __repr__(self):
344
+ return (
345
+ f"sqlalchemy.testing.config.Config"
346
+ f"({self.db.name}+{self.db.driver}, "
347
+ f"{self.db.dialect.server_version_info})"
348
+ )
349
+
350
+ def _set_name(self, db):
351
+ suffix = "_async" if db.dialect.is_async else ""
352
+ if db.dialect.server_version_info:
353
+ svi = ".".join(str(tok) for tok in db.dialect.server_version_info)
354
+ self.name = "%s+%s%s_[%s]" % (db.name, db.driver, suffix, svi)
355
+ else:
356
+ self.name = "%s+%s%s" % (db.name, db.driver, suffix)
357
+
358
+ @classmethod
359
+ def register(cls, db, db_opts, options, file_config):
360
+ """add a config as one of the global configs.
361
+
362
+ If there are no configs set up yet, this config also
363
+ gets set as the "_current".
364
+ """
365
+ global any_async
366
+
367
+ cfg = Config(db, db_opts, options, file_config)
368
+
369
+ # if any backends include an async driver, then ensure
370
+ # all setup/teardown and tests are wrapped in the maybe_async()
371
+ # decorator that will set up a greenlet context for async drivers.
372
+ any_async = any_async or cfg.is_async
373
+
374
+ cls._configs.add(cfg)
375
+ return cfg
376
+
377
+ @classmethod
378
+ def set_as_current(cls, config, namespace):
379
+ global db, _current, db_url, test_schema, test_schema_2, db_opts
380
+ _current = config
381
+ db_url = config.db.url
382
+ db_opts = config.db_opts
383
+ test_schema = config.test_schema
384
+ test_schema_2 = config.test_schema_2
385
+ namespace.db = db = config.db
386
+
387
+ @classmethod
388
+ def push_engine(cls, db, namespace):
389
+ assert _current, "Can't push without a default Config set up"
390
+ cls.push(
391
+ Config(
392
+ db, _current.db_opts, _current.options, _current.file_config
393
+ ),
394
+ namespace,
395
+ )
396
+
397
+ @classmethod
398
+ def push(cls, config, namespace):
399
+ cls._stack.append(_current)
400
+ cls.set_as_current(config, namespace)
401
+
402
+ @classmethod
403
+ def pop(cls, namespace):
404
+ if cls._stack:
405
+ # a failed test w/ -x option can call reset() ahead of time
406
+ _current = cls._stack[-1]
407
+ del cls._stack[-1]
408
+ cls.set_as_current(_current, namespace)
409
+
410
+ @classmethod
411
+ def reset(cls, namespace):
412
+ if cls._stack:
413
+ cls.set_as_current(cls._stack[0], namespace)
414
+ cls._stack.clear()
415
+
416
+ @classmethod
417
+ def all_configs(cls):
418
+ return cls._configs
419
+
420
+ @classmethod
421
+ def all_dbs(cls):
422
+ for cfg in cls.all_configs():
423
+ yield cfg.db
424
+
425
+ def skip_test(self, msg):
426
+ skip_test(msg)
427
+
428
+
429
+ def skip_test(msg):
430
+ raise _fixture_functions.skip_test_exception(msg)
431
+
432
+
433
+ def async_test(fn):
434
+ return _fixture_functions.async_test(fn)