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
@@ -0,0 +1,674 @@
1
+ # event/attr.py
2
+ # Copyright (C) 2005-2026 the SQLAlchemy authors and contributors
3
+ # <see AUTHORS file>
4
+ #
5
+ # This module is part of SQLAlchemy and is released under
6
+ # the MIT License: https://www.opensource.org/licenses/mit-license.php
7
+
8
+ """Attribute implementation for _Dispatch classes.
9
+
10
+ The various listener targets for a particular event class are represented
11
+ as attributes, which refer to collections of listeners to be fired off.
12
+ These collections can exist at the class level as well as at the instance
13
+ level. An event is fired off using code like this::
14
+
15
+ some_object.dispatch.first_connect(arg1, arg2)
16
+
17
+ Above, ``some_object.dispatch`` would be an instance of ``_Dispatch`` and
18
+ ``first_connect`` is typically an instance of ``_ListenerCollection``
19
+ if event listeners are present, or ``_EmptyListener`` if none are present.
20
+
21
+ The attribute mechanics here spend effort trying to ensure listener functions
22
+ are available with a minimum of function call overhead, that unnecessary
23
+ objects aren't created (i.e. many empty per-instance listener collections),
24
+ as well as that everything is garbage collectable when owning references are
25
+ lost. Other features such as "propagation" of listener functions across
26
+ many ``_Dispatch`` instances, "joining" of multiple ``_Dispatch`` instances,
27
+ as well as support for subclass propagation (e.g. events assigned to
28
+ ``Pool`` vs. ``QueuePool``) are all implemented here.
29
+
30
+ """
31
+ from __future__ import annotations
32
+
33
+ import collections
34
+ from itertools import chain
35
+ import threading
36
+ from types import TracebackType
37
+ import typing
38
+ from typing import Any
39
+ from typing import cast
40
+ from typing import Collection
41
+ from typing import Deque
42
+ from typing import FrozenSet
43
+ from typing import Generic
44
+ from typing import Iterator
45
+ from typing import MutableMapping
46
+ from typing import MutableSequence
47
+ from typing import NoReturn
48
+ from typing import Optional
49
+ from typing import Protocol
50
+ from typing import Sequence
51
+ from typing import Set
52
+ from typing import Tuple
53
+ from typing import Type
54
+ from typing import TypeVar
55
+ from typing import Union
56
+ import weakref
57
+
58
+ from . import legacy
59
+ from . import registry
60
+ from .registry import _ET
61
+ from .registry import _EventKey
62
+ from .registry import _ListenerFnType
63
+ from .. import exc
64
+ from .. import util
65
+ from ..util.concurrency import AsyncAdaptedLock
66
+
67
+ _T = TypeVar("_T", bound=Any)
68
+
69
+ if typing.TYPE_CHECKING:
70
+ from .base import _Dispatch
71
+ from .base import _DispatchCommon
72
+ from .base import _HasEventsDispatch
73
+
74
+
75
+ class RefCollection(util.MemoizedSlots, Generic[_ET]):
76
+ __slots__ = ("ref",)
77
+
78
+ ref: weakref.ref[RefCollection[_ET]]
79
+
80
+ def _memoized_attr_ref(self) -> weakref.ref[RefCollection[_ET]]:
81
+ return weakref.ref(self, registry._collection_gced)
82
+
83
+
84
+ class _empty_collection(Collection[_T]):
85
+ def append(self, element: _T) -> None:
86
+ pass
87
+
88
+ def appendleft(self, element: _T) -> None:
89
+ pass
90
+
91
+ def extend(self, other: Sequence[_T]) -> None:
92
+ pass
93
+
94
+ def remove(self, element: _T) -> None:
95
+ pass
96
+
97
+ def __contains__(self, element: Any) -> bool:
98
+ return False
99
+
100
+ def __iter__(self) -> Iterator[_T]:
101
+ return iter([])
102
+
103
+ def clear(self) -> None:
104
+ pass
105
+
106
+ def __len__(self) -> int:
107
+ return 0
108
+
109
+
110
+ _ListenerFnSequenceType = Union[Deque[_T], _empty_collection[_T]]
111
+
112
+
113
+ class _ClsLevelDispatch(RefCollection[_ET]):
114
+ """Class-level events on :class:`._Dispatch` classes."""
115
+
116
+ __slots__ = (
117
+ "clsname",
118
+ "name",
119
+ "arg_names",
120
+ "has_kw",
121
+ "legacy_signatures",
122
+ "_clslevel",
123
+ "__weakref__",
124
+ )
125
+
126
+ clsname: str
127
+ name: str
128
+ arg_names: Sequence[str]
129
+ has_kw: bool
130
+ legacy_signatures: MutableSequence[legacy._LegacySignatureType]
131
+ _clslevel: MutableMapping[
132
+ Type[_ET], _ListenerFnSequenceType[_ListenerFnType]
133
+ ]
134
+
135
+ def __init__(
136
+ self,
137
+ parent_dispatch_cls: Type[_HasEventsDispatch[_ET]],
138
+ fn: _ListenerFnType,
139
+ ):
140
+ self.name = fn.__name__
141
+ self.clsname = parent_dispatch_cls.__name__
142
+ argspec = util.inspect_getfullargspec(fn)
143
+ self.arg_names = argspec.args[1:]
144
+ self.has_kw = bool(argspec.varkw)
145
+ self.legacy_signatures = list(
146
+ reversed(
147
+ sorted(
148
+ getattr(fn, "_legacy_signatures", []), key=lambda s: s[0]
149
+ )
150
+ )
151
+ )
152
+ fn.__doc__ = legacy._augment_fn_docs(self, parent_dispatch_cls, fn)
153
+
154
+ self._clslevel = weakref.WeakKeyDictionary()
155
+
156
+ def _adjust_fn_spec(
157
+ self, fn: _ListenerFnType, named: bool
158
+ ) -> _ListenerFnType:
159
+ if named:
160
+ fn = self._wrap_fn_for_kw(fn)
161
+ if self.legacy_signatures:
162
+ try:
163
+ argspec = util.get_callable_argspec(fn, no_self=True)
164
+ except TypeError:
165
+ pass
166
+ else:
167
+ fn = legacy._wrap_fn_for_legacy(self, fn, argspec)
168
+ return fn
169
+
170
+ def _wrap_fn_for_kw(self, fn: _ListenerFnType) -> _ListenerFnType:
171
+ def wrap_kw(*args: Any, **kw: Any) -> Any:
172
+ argdict = dict(zip(self.arg_names, args))
173
+ argdict.update(kw)
174
+ return fn(**argdict)
175
+
176
+ return wrap_kw
177
+
178
+ def _do_insert_or_append(
179
+ self, event_key: _EventKey[_ET], is_append: bool
180
+ ) -> None:
181
+ target = event_key.dispatch_target
182
+ assert isinstance(
183
+ target, type
184
+ ), "Class-level Event targets must be classes."
185
+ if not getattr(target, "_sa_propagate_class_events", True):
186
+ raise exc.InvalidRequestError(
187
+ f"Can't assign an event directly to the {target} class"
188
+ )
189
+
190
+ cls: Type[_ET]
191
+
192
+ for cls in util.walk_subclasses(target):
193
+ if cls is not target and cls not in self._clslevel:
194
+ self.update_subclass(cls)
195
+ else:
196
+ if cls not in self._clslevel:
197
+ self.update_subclass(cls)
198
+ if is_append:
199
+ self._clslevel[cls].append(event_key._listen_fn)
200
+ else:
201
+ self._clslevel[cls].appendleft(event_key._listen_fn)
202
+ registry._stored_in_collection(event_key, self)
203
+
204
+ def insert(self, event_key: _EventKey[_ET], propagate: bool) -> None:
205
+ self._do_insert_or_append(event_key, is_append=False)
206
+
207
+ def append(self, event_key: _EventKey[_ET], propagate: bool) -> None:
208
+ self._do_insert_or_append(event_key, is_append=True)
209
+
210
+ def update_subclass(self, target: Type[_ET]) -> None:
211
+ if target not in self._clslevel:
212
+ if getattr(target, "_sa_propagate_class_events", True):
213
+ self._clslevel[target] = collections.deque()
214
+ else:
215
+ self._clslevel[target] = _empty_collection()
216
+
217
+ clslevel = self._clslevel[target]
218
+ cls: Type[_ET]
219
+ for cls in target.__mro__[1:]:
220
+ if cls in self._clslevel:
221
+ clslevel.extend(
222
+ [fn for fn in self._clslevel[cls] if fn not in clslevel]
223
+ )
224
+
225
+ def remove(self, event_key: _EventKey[_ET]) -> None:
226
+ target = event_key.dispatch_target
227
+ cls: Type[_ET]
228
+ for cls in util.walk_subclasses(target):
229
+ if cls in self._clslevel:
230
+ self._clslevel[cls].remove(event_key._listen_fn)
231
+ registry._removed_from_collection(event_key, self)
232
+
233
+ def clear(self) -> None:
234
+ """Clear all class level listeners"""
235
+
236
+ to_clear: Set[_ListenerFnType] = set()
237
+ for dispatcher in self._clslevel.values():
238
+ to_clear.update(dispatcher)
239
+ dispatcher.clear()
240
+ registry._clear(self, to_clear)
241
+
242
+ def for_modify(self, obj: _Dispatch[_ET]) -> _ClsLevelDispatch[_ET]:
243
+ """Return an event collection which can be modified.
244
+
245
+ For _ClsLevelDispatch at the class level of
246
+ a dispatcher, this returns self.
247
+
248
+ """
249
+ return self
250
+
251
+
252
+ class _InstanceLevelDispatch(RefCollection[_ET], Collection[_ListenerFnType]):
253
+ __slots__ = ()
254
+
255
+ parent: _ClsLevelDispatch[_ET]
256
+
257
+ def _adjust_fn_spec(
258
+ self, fn: _ListenerFnType, named: bool
259
+ ) -> _ListenerFnType:
260
+ return self.parent._adjust_fn_spec(fn, named)
261
+
262
+ def __contains__(self, item: Any) -> bool:
263
+ raise NotImplementedError()
264
+
265
+ def __len__(self) -> int:
266
+ raise NotImplementedError()
267
+
268
+ def __iter__(self) -> Iterator[_ListenerFnType]:
269
+ raise NotImplementedError()
270
+
271
+ def __bool__(self) -> bool:
272
+ raise NotImplementedError()
273
+
274
+ def exec_once(self, *args: Any, **kw: Any) -> None:
275
+ raise NotImplementedError()
276
+
277
+ def exec_once_unless_exception(self, *args: Any, **kw: Any) -> None:
278
+ raise NotImplementedError()
279
+
280
+ def _exec_w_sync_on_first_run(self, *args: Any, **kw: Any) -> None:
281
+ raise NotImplementedError()
282
+
283
+ def __call__(self, *args: Any, **kw: Any) -> None:
284
+ raise NotImplementedError()
285
+
286
+ def insert(self, event_key: _EventKey[_ET], propagate: bool) -> None:
287
+ raise NotImplementedError()
288
+
289
+ def append(self, event_key: _EventKey[_ET], propagate: bool) -> None:
290
+ raise NotImplementedError()
291
+
292
+ def remove(self, event_key: _EventKey[_ET]) -> None:
293
+ raise NotImplementedError()
294
+
295
+ def for_modify(
296
+ self, obj: _DispatchCommon[_ET]
297
+ ) -> _InstanceLevelDispatch[_ET]:
298
+ """Return an event collection which can be modified.
299
+
300
+ For _ClsLevelDispatch at the class level of
301
+ a dispatcher, this returns self.
302
+
303
+ """
304
+ return self
305
+
306
+
307
+ class _EmptyListener(_InstanceLevelDispatch[_ET]):
308
+ """Serves as a proxy interface to the events
309
+ served by a _ClsLevelDispatch, when there are no
310
+ instance-level events present.
311
+
312
+ Is replaced by _ListenerCollection when instance-level
313
+ events are added.
314
+
315
+ """
316
+
317
+ __slots__ = "parent", "parent_listeners", "name"
318
+
319
+ propagate: FrozenSet[_ListenerFnType] = frozenset()
320
+ listeners: Tuple[()] = ()
321
+ parent: _ClsLevelDispatch[_ET]
322
+ parent_listeners: _ListenerFnSequenceType[_ListenerFnType]
323
+ name: str
324
+
325
+ def __init__(self, parent: _ClsLevelDispatch[_ET], target_cls: Type[_ET]):
326
+ if target_cls not in parent._clslevel:
327
+ parent.update_subclass(target_cls)
328
+ self.parent = parent
329
+ self.parent_listeners = parent._clslevel[target_cls]
330
+ self.name = parent.name
331
+
332
+ def for_modify(
333
+ self, obj: _DispatchCommon[_ET]
334
+ ) -> _ListenerCollection[_ET]:
335
+ """Return an event collection which can be modified.
336
+
337
+ For _EmptyListener at the instance level of
338
+ a dispatcher, this generates a new
339
+ _ListenerCollection, applies it to the instance,
340
+ and returns it.
341
+
342
+ """
343
+ obj = cast("_Dispatch[_ET]", obj)
344
+
345
+ assert obj._instance_cls is not None
346
+ existing = getattr(obj, self.name)
347
+
348
+ with util.mini_gil:
349
+ if existing is self or isinstance(existing, _JoinedListener):
350
+ result = _ListenerCollection(self.parent, obj._instance_cls)
351
+ else:
352
+ # this codepath is an extremely rare race condition
353
+ # that has been observed in test_pool.py->test_timeout_race
354
+ # with freethreaded.
355
+ assert isinstance(existing, _ListenerCollection)
356
+ return existing
357
+
358
+ if existing is self:
359
+ setattr(obj, self.name, result)
360
+ return result
361
+
362
+ def _needs_modify(self, *args: Any, **kw: Any) -> NoReturn:
363
+ raise NotImplementedError("need to call for_modify()")
364
+
365
+ def exec_once(self, *args: Any, **kw: Any) -> NoReturn:
366
+ self._needs_modify(*args, **kw)
367
+
368
+ def exec_once_unless_exception(self, *args: Any, **kw: Any) -> NoReturn:
369
+ self._needs_modify(*args, **kw)
370
+
371
+ def insert(self, *args: Any, **kw: Any) -> NoReturn:
372
+ self._needs_modify(*args, **kw)
373
+
374
+ def append(self, *args: Any, **kw: Any) -> NoReturn:
375
+ self._needs_modify(*args, **kw)
376
+
377
+ def remove(self, *args: Any, **kw: Any) -> NoReturn:
378
+ self._needs_modify(*args, **kw)
379
+
380
+ def clear(self, *args: Any, **kw: Any) -> NoReturn:
381
+ self._needs_modify(*args, **kw)
382
+
383
+ def __call__(self, *args: Any, **kw: Any) -> None:
384
+ """Execute this event."""
385
+
386
+ for fn in self.parent_listeners:
387
+ fn(*args, **kw)
388
+
389
+ def __contains__(self, item: Any) -> bool:
390
+ return item in self.parent_listeners
391
+
392
+ def __len__(self) -> int:
393
+ return len(self.parent_listeners)
394
+
395
+ def __iter__(self) -> Iterator[_ListenerFnType]:
396
+ return iter(self.parent_listeners)
397
+
398
+ def __bool__(self) -> bool:
399
+ return bool(self.parent_listeners)
400
+
401
+
402
+ class _MutexProtocol(Protocol):
403
+ def __enter__(self) -> bool: ...
404
+
405
+ def __exit__(
406
+ self,
407
+ exc_type: Optional[Type[BaseException]],
408
+ exc_val: Optional[BaseException],
409
+ exc_tb: Optional[TracebackType],
410
+ ) -> Optional[bool]: ...
411
+
412
+
413
+ class _CompoundListener(_InstanceLevelDispatch[_ET]):
414
+ __slots__ = (
415
+ "_exec_once_mutex",
416
+ "_exec_once",
417
+ "_exec_w_sync_once",
418
+ "_is_asyncio",
419
+ )
420
+
421
+ _exec_once_mutex: Optional[_MutexProtocol]
422
+ parent_listeners: Collection[_ListenerFnType]
423
+ listeners: Collection[_ListenerFnType]
424
+ _exec_once: bool
425
+ _exec_w_sync_once: bool
426
+
427
+ def __init__(self, *arg: Any, **kw: Any):
428
+ super().__init__(*arg, **kw)
429
+ self._is_asyncio = False
430
+
431
+ def _set_asyncio(self) -> None:
432
+ self._is_asyncio = True
433
+
434
+ def _get_exec_once_mutex(self) -> _MutexProtocol:
435
+ with util.mini_gil:
436
+ if self._exec_once_mutex is not None:
437
+ return self._exec_once_mutex
438
+
439
+ if self._is_asyncio:
440
+ mutex = AsyncAdaptedLock()
441
+ else:
442
+ mutex = threading.Lock() # type: ignore[assignment]
443
+ self._exec_once_mutex = mutex
444
+
445
+ return mutex
446
+
447
+ def _exec_once_impl(
448
+ self, retry_on_exception: bool, *args: Any, **kw: Any
449
+ ) -> None:
450
+ with self._get_exec_once_mutex():
451
+ if not self._exec_once:
452
+ try:
453
+ self(*args, **kw)
454
+ exception = False
455
+ except:
456
+ exception = True
457
+ raise
458
+ finally:
459
+ if not exception or not retry_on_exception:
460
+ self._exec_once = True
461
+
462
+ def exec_once(self, *args: Any, **kw: Any) -> None:
463
+ """Execute this event, but only if it has not been
464
+ executed already for this collection."""
465
+
466
+ if not self._exec_once:
467
+ self._exec_once_impl(False, *args, **kw)
468
+
469
+ def exec_once_unless_exception(self, *args: Any, **kw: Any) -> None:
470
+ """Execute this event, but only if it has not been
471
+ executed already for this collection, or was called
472
+ by a previous exec_once_unless_exception call and
473
+ raised an exception.
474
+
475
+ If exec_once was already called, then this method will never run
476
+ the callable regardless of whether it raised or not.
477
+
478
+ """
479
+ if not self._exec_once:
480
+ self._exec_once_impl(True, *args, **kw)
481
+
482
+ def _exec_w_sync_on_first_run(self, *args: Any, **kw: Any) -> None:
483
+ """Execute this event, and use a mutex if it has not been
484
+ executed already for this collection, or was called
485
+ by a previous _exec_w_sync_on_first_run call and
486
+ raised an exception.
487
+
488
+ If _exec_w_sync_on_first_run was already called and didn't raise an
489
+ exception, then a mutex is not used. It's not guaranteed
490
+ the mutex won't be used more than once in the case of very rare
491
+ race conditions.
492
+
493
+ .. versionadded:: 1.4.11
494
+
495
+ """
496
+ if not self._exec_w_sync_once:
497
+ with self._get_exec_once_mutex():
498
+ try:
499
+ self(*args, **kw)
500
+ except:
501
+ raise
502
+ else:
503
+ self._exec_w_sync_once = True
504
+ else:
505
+ self(*args, **kw)
506
+
507
+ def __call__(self, *args: Any, **kw: Any) -> None:
508
+ """Execute this event."""
509
+
510
+ for fn in self.parent_listeners:
511
+ fn(*args, **kw)
512
+ for fn in self.listeners:
513
+ fn(*args, **kw)
514
+
515
+ def __contains__(self, item: Any) -> bool:
516
+ return item in self.parent_listeners or item in self.listeners
517
+
518
+ def __len__(self) -> int:
519
+ return len(self.parent_listeners) + len(self.listeners)
520
+
521
+ def __iter__(self) -> Iterator[_ListenerFnType]:
522
+ return chain(self.parent_listeners, self.listeners)
523
+
524
+ def __bool__(self) -> bool:
525
+ return bool(self.listeners or self.parent_listeners)
526
+
527
+
528
+ class _ListenerCollection(_CompoundListener[_ET]):
529
+ """Instance-level attributes on instances of :class:`._Dispatch`.
530
+
531
+ Represents a collection of listeners.
532
+
533
+ As of 0.7.9, _ListenerCollection is only first
534
+ created via the _EmptyListener.for_modify() method.
535
+
536
+ """
537
+
538
+ __slots__ = (
539
+ "parent_listeners",
540
+ "parent",
541
+ "name",
542
+ "listeners",
543
+ "propagate",
544
+ "__weakref__",
545
+ )
546
+
547
+ parent_listeners: Collection[_ListenerFnType]
548
+ parent: _ClsLevelDispatch[_ET]
549
+ name: str
550
+ listeners: Deque[_ListenerFnType]
551
+ propagate: Set[_ListenerFnType]
552
+
553
+ def __init__(self, parent: _ClsLevelDispatch[_ET], target_cls: Type[_ET]):
554
+ super().__init__()
555
+ if target_cls not in parent._clslevel:
556
+ parent.update_subclass(target_cls)
557
+ self._exec_once = False
558
+ self._exec_w_sync_once = False
559
+ self._exec_once_mutex = None
560
+ self.parent_listeners = parent._clslevel[target_cls]
561
+ self.parent = parent
562
+ self.name = parent.name
563
+ self.listeners = collections.deque()
564
+ self.propagate = set()
565
+
566
+ def for_modify(
567
+ self, obj: _DispatchCommon[_ET]
568
+ ) -> _ListenerCollection[_ET]:
569
+ """Return an event collection which can be modified.
570
+
571
+ For _ListenerCollection at the instance level of
572
+ a dispatcher, this returns self.
573
+
574
+ """
575
+ return self
576
+
577
+ def _update(
578
+ self, other: _ListenerCollection[_ET], only_propagate: bool = True
579
+ ) -> None:
580
+ """Populate from the listeners in another :class:`_Dispatch`
581
+ object."""
582
+ existing_listeners = self.listeners
583
+ existing_listener_set = set(existing_listeners)
584
+ self.propagate.update(other.propagate)
585
+ other_listeners = [
586
+ l
587
+ for l in other.listeners
588
+ if l not in existing_listener_set
589
+ and not only_propagate
590
+ or l in self.propagate
591
+ ]
592
+
593
+ existing_listeners.extend(other_listeners)
594
+
595
+ if other._is_asyncio:
596
+ self._set_asyncio()
597
+
598
+ to_associate = other.propagate.union(other_listeners)
599
+ registry._stored_in_collection_multi(self, other, to_associate)
600
+
601
+ def insert(self, event_key: _EventKey[_ET], propagate: bool) -> None:
602
+ if event_key.prepend_to_list(self, self.listeners):
603
+ if propagate:
604
+ self.propagate.add(event_key._listen_fn)
605
+
606
+ def append(self, event_key: _EventKey[_ET], propagate: bool) -> None:
607
+ if event_key.append_to_list(self, self.listeners):
608
+ if propagate:
609
+ self.propagate.add(event_key._listen_fn)
610
+
611
+ def remove(self, event_key: _EventKey[_ET]) -> None:
612
+ self.listeners.remove(event_key._listen_fn)
613
+ self.propagate.discard(event_key._listen_fn)
614
+ registry._removed_from_collection(event_key, self)
615
+
616
+ def clear(self) -> None:
617
+ registry._clear(self, self.listeners)
618
+ self.propagate.clear()
619
+ self.listeners.clear()
620
+
621
+
622
+ class _JoinedListener(_CompoundListener[_ET]):
623
+ __slots__ = "parent_dispatch", "name", "local", "parent_listeners"
624
+
625
+ parent_dispatch: _DispatchCommon[_ET]
626
+ name: str
627
+ local: _InstanceLevelDispatch[_ET]
628
+ parent_listeners: Collection[_ListenerFnType]
629
+
630
+ def __init__(
631
+ self,
632
+ parent_dispatch: _DispatchCommon[_ET],
633
+ name: str,
634
+ local: _EmptyListener[_ET],
635
+ ):
636
+ self._exec_once = False
637
+ self._exec_w_sync_once = False
638
+ self._exec_once_mutex = None
639
+ self.parent_dispatch = parent_dispatch
640
+ self.name = name
641
+ self.local = local
642
+ self.parent_listeners = self.local
643
+
644
+ if not typing.TYPE_CHECKING:
645
+ # first error, I don't really understand:
646
+ # Signature of "listeners" incompatible with
647
+ # supertype "_CompoundListener" [override]
648
+ # the name / return type are exactly the same
649
+ # second error is getattr_isn't typed, the cast() here
650
+ # adds too much method overhead
651
+ @property
652
+ def listeners(self) -> Collection[_ListenerFnType]:
653
+ return getattr(self.parent_dispatch, self.name)
654
+
655
+ def _adjust_fn_spec(
656
+ self, fn: _ListenerFnType, named: bool
657
+ ) -> _ListenerFnType:
658
+ return self.local._adjust_fn_spec(fn, named)
659
+
660
+ def for_modify(self, obj: _DispatchCommon[_ET]) -> _JoinedListener[_ET]:
661
+ self.local = self.parent_listeners = self.local.for_modify(obj)
662
+ return self
663
+
664
+ def insert(self, event_key: _EventKey[_ET], propagate: bool) -> None:
665
+ self.local.insert(event_key, propagate)
666
+
667
+ def append(self, event_key: _EventKey[_ET], propagate: bool) -> None:
668
+ self.local.append(event_key, propagate)
669
+
670
+ def remove(self, event_key: _EventKey[_ET]) -> None:
671
+ self.local.remove(event_key)
672
+
673
+ def clear(self) -> None:
674
+ raise NotImplementedError()