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,712 @@
1
+ # util/_collections.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
+ """Collection classes and helpers."""
10
+ from __future__ import annotations
11
+
12
+ import operator
13
+ import threading
14
+ import types
15
+ import typing
16
+ from typing import Any
17
+ from typing import Callable
18
+ from typing import cast
19
+ from typing import Container
20
+ from typing import Dict
21
+ from typing import FrozenSet
22
+ from typing import Generic
23
+ from typing import Iterable
24
+ from typing import Iterator
25
+ from typing import List
26
+ from typing import Mapping
27
+ from typing import NoReturn
28
+ from typing import Optional
29
+ from typing import overload
30
+ from typing import Sequence
31
+ from typing import Set
32
+ from typing import Tuple
33
+ from typing import TypeVar
34
+ from typing import Union
35
+ from typing import ValuesView
36
+ import weakref
37
+
38
+ from ._has_cy import HAS_CYEXTENSION
39
+ from .typing import is_non_string_iterable
40
+ from .typing import Literal
41
+ from .typing import Protocol
42
+
43
+ if typing.TYPE_CHECKING or not HAS_CYEXTENSION:
44
+ from ._py_collections import immutabledict as immutabledict
45
+ from ._py_collections import IdentitySet as IdentitySet
46
+ from ._py_collections import ReadOnlyContainer as ReadOnlyContainer
47
+ from ._py_collections import ImmutableDictBase as ImmutableDictBase
48
+ from ._py_collections import OrderedSet as OrderedSet
49
+ from ._py_collections import unique_list as unique_list
50
+ else:
51
+ from sqlalchemy.cyextension.immutabledict import (
52
+ ReadOnlyContainer as ReadOnlyContainer,
53
+ )
54
+ from sqlalchemy.cyextension.immutabledict import (
55
+ ImmutableDictBase as ImmutableDictBase,
56
+ )
57
+ from sqlalchemy.cyextension.immutabledict import (
58
+ immutabledict as immutabledict,
59
+ )
60
+ from sqlalchemy.cyextension.collections import IdentitySet as IdentitySet
61
+ from sqlalchemy.cyextension.collections import OrderedSet as OrderedSet
62
+ from sqlalchemy.cyextension.collections import ( # noqa
63
+ unique_list as unique_list,
64
+ )
65
+
66
+
67
+ _T = TypeVar("_T", bound=Any)
68
+ _KT = TypeVar("_KT", bound=Any)
69
+ _VT = TypeVar("_VT", bound=Any)
70
+ _T_co = TypeVar("_T_co", covariant=True)
71
+
72
+ EMPTY_SET: FrozenSet[Any] = frozenset()
73
+ NONE_SET: FrozenSet[Any] = frozenset([None])
74
+
75
+
76
+ def merge_lists_w_ordering(a: List[Any], b: List[Any]) -> List[Any]:
77
+ """merge two lists, maintaining ordering as much as possible.
78
+
79
+ this is to reconcile vars(cls) with cls.__annotations__.
80
+
81
+ Example::
82
+
83
+ >>> a = ["__tablename__", "id", "x", "created_at"]
84
+ >>> b = ["id", "name", "data", "y", "created_at"]
85
+ >>> merge_lists_w_ordering(a, b)
86
+ ['__tablename__', 'id', 'name', 'data', 'y', 'x', 'created_at']
87
+
88
+ This is not necessarily the ordering that things had on the class,
89
+ in this case the class is::
90
+
91
+ class User(Base):
92
+ __tablename__ = "users"
93
+
94
+ id: Mapped[int] = mapped_column(primary_key=True)
95
+ name: Mapped[str]
96
+ data: Mapped[Optional[str]]
97
+ x = Column(Integer)
98
+ y: Mapped[int]
99
+ created_at: Mapped[datetime.datetime] = mapped_column()
100
+
101
+ But things are *mostly* ordered.
102
+
103
+ The algorithm could also be done by creating a partial ordering for
104
+ all items in both lists and then using topological_sort(), but that
105
+ is too much overhead.
106
+
107
+ Background on how I came up with this is at:
108
+ https://gist.github.com/zzzeek/89de958cf0803d148e74861bd682ebae
109
+
110
+ """
111
+ overlap = set(a).intersection(b)
112
+
113
+ result = []
114
+
115
+ current, other = iter(a), iter(b)
116
+
117
+ while True:
118
+ for element in current:
119
+ if element in overlap:
120
+ overlap.discard(element)
121
+ other, current = current, other
122
+ break
123
+
124
+ result.append(element)
125
+ else:
126
+ result.extend(other)
127
+ break
128
+
129
+ return result
130
+
131
+
132
+ def coerce_to_immutabledict(d: Mapping[_KT, _VT]) -> immutabledict[_KT, _VT]:
133
+ if not d:
134
+ return EMPTY_DICT
135
+ elif isinstance(d, immutabledict):
136
+ return d
137
+ else:
138
+ return immutabledict(d)
139
+
140
+
141
+ EMPTY_DICT: immutabledict[Any, Any] = immutabledict()
142
+
143
+
144
+ class FacadeDict(ImmutableDictBase[_KT, _VT]):
145
+ """A dictionary that is not publicly mutable."""
146
+
147
+ def __new__(cls, *args: Any) -> FacadeDict[Any, Any]:
148
+ new = ImmutableDictBase.__new__(cls)
149
+ return new
150
+
151
+ def copy(self) -> NoReturn:
152
+ raise NotImplementedError(
153
+ "an immutabledict shouldn't need to be copied. use dict(d) "
154
+ "if you need a mutable dictionary."
155
+ )
156
+
157
+ def __reduce__(self) -> Any:
158
+ return FacadeDict, (dict(self),)
159
+
160
+ def _insert_item(self, key: _KT, value: _VT) -> None:
161
+ """insert an item into the dictionary directly."""
162
+ dict.__setitem__(self, key, value)
163
+
164
+ def __repr__(self) -> str:
165
+ return "FacadeDict(%s)" % dict.__repr__(self)
166
+
167
+
168
+ _DT = TypeVar("_DT", bound=Any)
169
+
170
+ _F = TypeVar("_F", bound=Any)
171
+
172
+
173
+ class Properties(Generic[_T]):
174
+ """Provide a __getattr__/__setattr__ interface over a dict."""
175
+
176
+ __slots__ = ("_data",)
177
+
178
+ _data: Dict[str, _T]
179
+
180
+ def __init__(self, data: Dict[str, _T]):
181
+ object.__setattr__(self, "_data", data)
182
+
183
+ def __len__(self) -> int:
184
+ return len(self._data)
185
+
186
+ def __iter__(self) -> Iterator[_T]:
187
+ return iter(list(self._data.values()))
188
+
189
+ def __dir__(self) -> List[str]:
190
+ return dir(super()) + [str(k) for k in self._data.keys()]
191
+
192
+ def __add__(self, other: Properties[_F]) -> List[Union[_T, _F]]:
193
+ return list(self) + list(other)
194
+
195
+ def __setitem__(self, key: str, obj: _T) -> None:
196
+ self._data[key] = obj
197
+
198
+ def __getitem__(self, key: str) -> _T:
199
+ return self._data[key]
200
+
201
+ def __delitem__(self, key: str) -> None:
202
+ del self._data[key]
203
+
204
+ def __setattr__(self, key: str, obj: _T) -> None:
205
+ self._data[key] = obj
206
+
207
+ def __getstate__(self) -> Dict[str, Any]:
208
+ return {"_data": self._data}
209
+
210
+ def __setstate__(self, state: Dict[str, Any]) -> None:
211
+ object.__setattr__(self, "_data", state["_data"])
212
+
213
+ def __getattr__(self, key: str) -> _T:
214
+ try:
215
+ return self._data[key]
216
+ except KeyError:
217
+ raise AttributeError(key)
218
+
219
+ def __contains__(self, key: str) -> bool:
220
+ return key in self._data
221
+
222
+ def as_readonly(self) -> ReadOnlyProperties[_T]:
223
+ """Return an immutable proxy for this :class:`.Properties`."""
224
+
225
+ return ReadOnlyProperties(self._data)
226
+
227
+ def update(self, value: Dict[str, _T]) -> None:
228
+ self._data.update(value)
229
+
230
+ @overload
231
+ def get(self, key: str) -> Optional[_T]: ...
232
+
233
+ @overload
234
+ def get(self, key: str, default: Union[_DT, _T]) -> Union[_DT, _T]: ...
235
+
236
+ def get(
237
+ self, key: str, default: Optional[Union[_DT, _T]] = None
238
+ ) -> Optional[Union[_T, _DT]]:
239
+ if key in self:
240
+ return self[key]
241
+ else:
242
+ return default
243
+
244
+ def keys(self) -> List[str]:
245
+ return list(self._data)
246
+
247
+ def values(self) -> List[_T]:
248
+ return list(self._data.values())
249
+
250
+ def items(self) -> List[Tuple[str, _T]]:
251
+ return list(self._data.items())
252
+
253
+ def has_key(self, key: str) -> bool:
254
+ return key in self._data
255
+
256
+ def clear(self) -> None:
257
+ self._data.clear()
258
+
259
+
260
+ class OrderedProperties(Properties[_T]):
261
+ """Provide a __getattr__/__setattr__ interface with an OrderedDict
262
+ as backing store."""
263
+
264
+ __slots__ = ()
265
+
266
+ def __init__(self):
267
+ Properties.__init__(self, OrderedDict())
268
+
269
+
270
+ class ReadOnlyProperties(ReadOnlyContainer, Properties[_T]):
271
+ """Provide immutable dict/object attribute to an underlying dictionary."""
272
+
273
+ __slots__ = ()
274
+
275
+
276
+ def _ordered_dictionary_sort(d, key=None):
277
+ """Sort an OrderedDict in-place."""
278
+
279
+ items = [(k, d[k]) for k in sorted(d, key=key)]
280
+
281
+ d.clear()
282
+
283
+ d.update(items)
284
+
285
+
286
+ OrderedDict = dict
287
+ sort_dictionary = _ordered_dictionary_sort
288
+
289
+
290
+ class WeakSequence(Sequence[_T]):
291
+ def __init__(self, __elements: Sequence[_T] = ()):
292
+ # adapted from weakref.WeakKeyDictionary, prevent reference
293
+ # cycles in the collection itself
294
+ def _remove(item, selfref=weakref.ref(self)):
295
+ self = selfref()
296
+ if self is not None:
297
+ self._storage.remove(item)
298
+
299
+ self._remove = _remove
300
+ self._storage = [
301
+ weakref.ref(element, _remove) for element in __elements
302
+ ]
303
+
304
+ def append(self, item):
305
+ self._storage.append(weakref.ref(item, self._remove))
306
+
307
+ def __len__(self):
308
+ return len(self._storage)
309
+
310
+ def __iter__(self):
311
+ return (
312
+ obj for obj in (ref() for ref in self._storage) if obj is not None
313
+ )
314
+
315
+ def __getitem__(self, index):
316
+ return self._storage[index]()
317
+
318
+
319
+ class OrderedIdentitySet(IdentitySet):
320
+ def __init__(self, iterable: Optional[Iterable[Any]] = None):
321
+ IdentitySet.__init__(self)
322
+ self._members = OrderedDict()
323
+ if iterable:
324
+ for o in iterable:
325
+ self.add(o)
326
+
327
+
328
+ class PopulateDict(Dict[_KT, _VT]):
329
+ """A dict which populates missing values via a creation function.
330
+
331
+ Note the creation function takes a key, unlike
332
+ collections.defaultdict.
333
+
334
+ """
335
+
336
+ def __init__(self, creator: Callable[[_KT], _VT]):
337
+ self.creator = creator
338
+
339
+ def __missing__(self, key: Any) -> Any:
340
+ self[key] = val = self.creator(key)
341
+ return val
342
+
343
+
344
+ class WeakPopulateDict(Dict[_KT, _VT]):
345
+ """Like PopulateDict, but assumes a self + a method and does not create
346
+ a reference cycle.
347
+
348
+ """
349
+
350
+ def __init__(self, creator_method: types.MethodType):
351
+ self.creator = creator_method.__func__
352
+ weakself = creator_method.__self__
353
+ self.weakself = weakref.ref(weakself)
354
+
355
+ def __missing__(self, key: Any) -> Any:
356
+ self[key] = val = self.creator(self.weakself(), key)
357
+ return val
358
+
359
+
360
+ # Define collections that are capable of storing
361
+ # ColumnElement objects as hashable keys/elements.
362
+ # At this point, these are mostly historical, things
363
+ # used to be more complicated.
364
+ column_set = set
365
+ column_dict = dict
366
+ ordered_column_set = OrderedSet
367
+
368
+
369
+ class UniqueAppender(Generic[_T]):
370
+ """Appends items to a collection ensuring uniqueness.
371
+
372
+ Additional appends() of the same object are ignored. Membership is
373
+ determined by identity (``is a``) not equality (``==``).
374
+ """
375
+
376
+ __slots__ = "data", "_data_appender", "_unique"
377
+
378
+ data: Union[Iterable[_T], Set[_T], List[_T]]
379
+ _data_appender: Callable[[_T], None]
380
+ _unique: Dict[int, Literal[True]]
381
+
382
+ def __init__(
383
+ self,
384
+ data: Union[Iterable[_T], Set[_T], List[_T]],
385
+ via: Optional[str] = None,
386
+ ):
387
+ self.data = data
388
+ self._unique = {}
389
+ if via:
390
+ self._data_appender = getattr(data, via)
391
+ elif hasattr(data, "append"):
392
+ self._data_appender = cast("List[_T]", data).append
393
+ elif hasattr(data, "add"):
394
+ self._data_appender = cast("Set[_T]", data).add
395
+
396
+ def append(self, item: _T) -> None:
397
+ id_ = id(item)
398
+ if id_ not in self._unique:
399
+ self._data_appender(item)
400
+ self._unique[id_] = True
401
+
402
+ def __iter__(self) -> Iterator[_T]:
403
+ return iter(self.data)
404
+
405
+
406
+ def coerce_generator_arg(arg: Any) -> List[Any]:
407
+ if len(arg) == 1 and isinstance(arg[0], types.GeneratorType):
408
+ return list(arg[0])
409
+ else:
410
+ return cast("List[Any]", arg)
411
+
412
+
413
+ def to_list(x: Any, default: Optional[List[Any]] = None) -> List[Any]:
414
+ if x is None:
415
+ return default # type: ignore
416
+ if not is_non_string_iterable(x):
417
+ return [x]
418
+ elif isinstance(x, list):
419
+ return x
420
+ else:
421
+ return list(x)
422
+
423
+
424
+ def has_intersection(set_: Container[Any], iterable: Iterable[Any]) -> bool:
425
+ r"""return True if any items of set\_ are present in iterable.
426
+
427
+ Goes through special effort to ensure __hash__ is not called
428
+ on items in iterable that don't support it.
429
+
430
+ """
431
+ return any(i in set_ for i in iterable if i.__hash__)
432
+
433
+
434
+ def to_set(x):
435
+ if x is None:
436
+ return set()
437
+ if not isinstance(x, set):
438
+ return set(to_list(x))
439
+ else:
440
+ return x
441
+
442
+
443
+ def to_column_set(x: Any) -> Set[Any]:
444
+ if x is None:
445
+ return column_set()
446
+ if not isinstance(x, column_set):
447
+ return column_set(to_list(x))
448
+ else:
449
+ return x
450
+
451
+
452
+ def update_copy(
453
+ d: Dict[Any, Any], _new: Optional[Dict[Any, Any]] = None, **kw: Any
454
+ ) -> Dict[Any, Any]:
455
+ """Copy the given dict and update with the given values."""
456
+
457
+ d = d.copy()
458
+ if _new:
459
+ d.update(_new)
460
+ d.update(**kw)
461
+ return d
462
+
463
+
464
+ def flatten_iterator(x: Iterable[_T]) -> Iterator[_T]:
465
+ """Given an iterator of which further sub-elements may also be
466
+ iterators, flatten the sub-elements into a single iterator.
467
+
468
+ """
469
+ elem: _T
470
+ for elem in x:
471
+ if not isinstance(elem, str) and hasattr(elem, "__iter__"):
472
+ yield from flatten_iterator(elem)
473
+ else:
474
+ yield elem
475
+
476
+
477
+ class LRUCache(typing.MutableMapping[_KT, _VT]):
478
+ """Dictionary with 'squishy' removal of least
479
+ recently used items.
480
+
481
+ Note that either get() or [] should be used here, but
482
+ generally its not safe to do an "in" check first as the dictionary
483
+ can change subsequent to that call.
484
+
485
+ """
486
+
487
+ __slots__ = (
488
+ "capacity",
489
+ "threshold",
490
+ "size_alert",
491
+ "_data",
492
+ "_counter",
493
+ "_mutex",
494
+ )
495
+
496
+ capacity: int
497
+ threshold: float
498
+ size_alert: Optional[Callable[[LRUCache[_KT, _VT]], None]]
499
+
500
+ def __init__(
501
+ self,
502
+ capacity: int = 100,
503
+ threshold: float = 0.5,
504
+ size_alert: Optional[Callable[..., None]] = None,
505
+ ):
506
+ self.capacity = capacity
507
+ self.threshold = threshold
508
+ self.size_alert = size_alert
509
+ self._counter = 0
510
+ self._mutex = threading.Lock()
511
+ self._data: Dict[_KT, Tuple[_KT, _VT, List[int]]] = {}
512
+
513
+ def _inc_counter(self):
514
+ self._counter += 1
515
+ return self._counter
516
+
517
+ @overload
518
+ def get(self, key: _KT) -> Optional[_VT]: ...
519
+
520
+ @overload
521
+ def get(self, key: _KT, default: Union[_VT, _T]) -> Union[_VT, _T]: ...
522
+
523
+ def get(
524
+ self, key: _KT, default: Optional[Union[_VT, _T]] = None
525
+ ) -> Optional[Union[_VT, _T]]:
526
+ item = self._data.get(key)
527
+ if item is not None:
528
+ item[2][0] = self._inc_counter()
529
+ return item[1]
530
+ else:
531
+ return default
532
+
533
+ def __getitem__(self, key: _KT) -> _VT:
534
+ item = self._data[key]
535
+ item[2][0] = self._inc_counter()
536
+ return item[1]
537
+
538
+ def __iter__(self) -> Iterator[_KT]:
539
+ return iter(self._data)
540
+
541
+ def __len__(self) -> int:
542
+ return len(self._data)
543
+
544
+ def values(self) -> ValuesView[_VT]:
545
+ return typing.ValuesView({k: i[1] for k, i in self._data.items()})
546
+
547
+ def __setitem__(self, key: _KT, value: _VT) -> None:
548
+ self._data[key] = (key, value, [self._inc_counter()])
549
+ self._manage_size()
550
+
551
+ def __delitem__(self, __v: _KT) -> None:
552
+ del self._data[__v]
553
+
554
+ @property
555
+ def size_threshold(self) -> float:
556
+ return self.capacity + self.capacity * self.threshold
557
+
558
+ def _manage_size(self) -> None:
559
+ if not self._mutex.acquire(False):
560
+ return
561
+ try:
562
+ size_alert = bool(self.size_alert)
563
+ while len(self) > self.capacity + self.capacity * self.threshold:
564
+ if size_alert:
565
+ size_alert = False
566
+ self.size_alert(self) # type: ignore
567
+ by_counter = sorted(
568
+ self._data.values(),
569
+ key=operator.itemgetter(2),
570
+ reverse=True,
571
+ )
572
+ for item in by_counter[self.capacity :]:
573
+ try:
574
+ del self._data[item[0]]
575
+ except KeyError:
576
+ # deleted elsewhere; skip
577
+ continue
578
+ finally:
579
+ self._mutex.release()
580
+
581
+
582
+ class _CreateFuncType(Protocol[_T_co]):
583
+ def __call__(self) -> _T_co: ...
584
+
585
+
586
+ class _ScopeFuncType(Protocol):
587
+ def __call__(self) -> Any: ...
588
+
589
+
590
+ class ScopedRegistry(Generic[_T]):
591
+ """A Registry that can store one or multiple instances of a single
592
+ class on the basis of a "scope" function.
593
+
594
+ The object implements ``__call__`` as the "getter", so by
595
+ calling ``myregistry()`` the contained object is returned
596
+ for the current scope.
597
+
598
+ :param createfunc:
599
+ a callable that returns a new object to be placed in the registry
600
+
601
+ :param scopefunc:
602
+ a callable that will return a key to store/retrieve an object.
603
+ """
604
+
605
+ __slots__ = "createfunc", "scopefunc", "registry"
606
+
607
+ createfunc: _CreateFuncType[_T]
608
+ scopefunc: _ScopeFuncType
609
+ registry: Any
610
+
611
+ def __init__(
612
+ self, createfunc: Callable[[], _T], scopefunc: Callable[[], Any]
613
+ ):
614
+ """Construct a new :class:`.ScopedRegistry`.
615
+
616
+ :param createfunc: A creation function that will generate
617
+ a new value for the current scope, if none is present.
618
+
619
+ :param scopefunc: A function that returns a hashable
620
+ token representing the current scope (such as, current
621
+ thread identifier).
622
+
623
+ """
624
+ self.createfunc = createfunc
625
+ self.scopefunc = scopefunc
626
+ self.registry = {}
627
+
628
+ def __call__(self) -> _T:
629
+ key = self.scopefunc()
630
+ try:
631
+ return self.registry[key] # type: ignore[no-any-return]
632
+ except KeyError:
633
+ return self.registry.setdefault(key, self.createfunc()) # type: ignore[no-any-return] # noqa: E501
634
+
635
+ def has(self) -> bool:
636
+ """Return True if an object is present in the current scope."""
637
+
638
+ return self.scopefunc() in self.registry
639
+
640
+ def set(self, obj: _T) -> None:
641
+ """Set the value for the current scope."""
642
+
643
+ self.registry[self.scopefunc()] = obj
644
+
645
+ def clear(self) -> None:
646
+ """Clear the current scope, if any."""
647
+
648
+ try:
649
+ del self.registry[self.scopefunc()]
650
+ except KeyError:
651
+ pass
652
+
653
+
654
+ class ThreadLocalRegistry(ScopedRegistry[_T]):
655
+ """A :class:`.ScopedRegistry` that uses a ``threading.local()``
656
+ variable for storage.
657
+
658
+ """
659
+
660
+ def __init__(self, createfunc: Callable[[], _T]):
661
+ self.createfunc = createfunc
662
+ self.registry = threading.local()
663
+
664
+ def __call__(self) -> _T:
665
+ try:
666
+ return self.registry.value # type: ignore[no-any-return]
667
+ except AttributeError:
668
+ val = self.registry.value = self.createfunc()
669
+ return val
670
+
671
+ def has(self) -> bool:
672
+ return hasattr(self.registry, "value")
673
+
674
+ def set(self, obj: _T) -> None:
675
+ self.registry.value = obj
676
+
677
+ def clear(self) -> None:
678
+ try:
679
+ del self.registry.value
680
+ except AttributeError:
681
+ pass
682
+
683
+
684
+ def has_dupes(sequence, target):
685
+ """Given a sequence and search object, return True if there's more
686
+ than one, False if zero or one of them.
687
+
688
+
689
+ """
690
+ # compare to .index version below, this version introduces less function
691
+ # overhead and is usually the same speed. At 15000 items (way bigger than
692
+ # a relationship-bound collection in memory usually is) it begins to
693
+ # fall behind the other version only by microseconds.
694
+ c = 0
695
+ for item in sequence:
696
+ if item is target:
697
+ c += 1
698
+ if c > 1:
699
+ return True
700
+ return False
701
+
702
+
703
+ # .index version. the two __contains__ calls as well
704
+ # as .index() and isinstance() slow this down.
705
+ # def has_dupes(sequence, target):
706
+ # if target not in sequence:
707
+ # return False
708
+ # elif not isinstance(sequence, collections_abc.Sequence):
709
+ # return False
710
+ #
711
+ # idx = sequence.index(target)
712
+ # return target in sequence[idx + 1:]