SQLAlchemy 2.0.47__cp313-cp313t-win_amd64.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-win_amd64.pyd +0 -0
  8. sqlalchemy/cyextension/collections.pyx +409 -0
  9. sqlalchemy/cyextension/immutabledict.cp313t-win_amd64.pyd +0 -0
  10. sqlalchemy/cyextension/immutabledict.pxd +8 -0
  11. sqlalchemy/cyextension/immutabledict.pyx +133 -0
  12. sqlalchemy/cyextension/processors.cp313t-win_amd64.pyd +0 -0
  13. sqlalchemy/cyextension/processors.pyx +68 -0
  14. sqlalchemy/cyextension/resultproxy.cp313t-win_amd64.pyd +0 -0
  15. sqlalchemy/cyextension/resultproxy.pyx +102 -0
  16. sqlalchemy/cyextension/util.cp313t-win_amd64.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,472 @@
1
+ # event/base.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
+ """Base implementation classes.
9
+
10
+ The public-facing ``Events`` serves as the base class for an event interface;
11
+ its public attributes represent different kinds of events. These attributes
12
+ are mirrored onto a ``_Dispatch`` class, which serves as a container for
13
+ collections of listener functions. These collections are represented both
14
+ at the class level of a particular ``_Dispatch`` class as well as within
15
+ instances of ``_Dispatch``.
16
+
17
+ """
18
+ from __future__ import annotations
19
+
20
+ import typing
21
+ from typing import Any
22
+ from typing import cast
23
+ from typing import Dict
24
+ from typing import Generic
25
+ from typing import Iterator
26
+ from typing import List
27
+ from typing import Mapping
28
+ from typing import MutableMapping
29
+ from typing import Optional
30
+ from typing import overload
31
+ from typing import Tuple
32
+ from typing import Type
33
+ from typing import Union
34
+ import weakref
35
+
36
+ from .attr import _ClsLevelDispatch
37
+ from .attr import _EmptyListener
38
+ from .attr import _InstanceLevelDispatch
39
+ from .attr import _JoinedListener
40
+ from .registry import _ET
41
+ from .registry import _EventKey
42
+ from .. import util
43
+ from ..util.typing import Literal
44
+
45
+ _registrars: MutableMapping[str, List[Type[_HasEventsDispatch[Any]]]] = (
46
+ util.defaultdict(list)
47
+ )
48
+
49
+
50
+ def _is_event_name(name: str) -> bool:
51
+ # _sa_event prefix is special to support internal-only event names.
52
+ # most event names are just plain method names that aren't
53
+ # underscored.
54
+
55
+ return (
56
+ not name.startswith("_") and name != "dispatch"
57
+ ) or name.startswith("_sa_event")
58
+
59
+
60
+ class _UnpickleDispatch:
61
+ """Serializable callable that re-generates an instance of
62
+ :class:`_Dispatch` given a particular :class:`.Events` subclass.
63
+
64
+ """
65
+
66
+ def __call__(self, _instance_cls: Type[_ET]) -> _Dispatch[_ET]:
67
+ for cls in _instance_cls.__mro__:
68
+ if "dispatch" in cls.__dict__:
69
+ return cast(
70
+ "_Dispatch[_ET]", cls.__dict__["dispatch"].dispatch
71
+ )._for_class(_instance_cls)
72
+ else:
73
+ raise AttributeError("No class with a 'dispatch' member present.")
74
+
75
+
76
+ class _DispatchCommon(Generic[_ET]):
77
+ __slots__ = ()
78
+
79
+ _instance_cls: Optional[Type[_ET]]
80
+
81
+ def _join(self, other: _DispatchCommon[_ET]) -> _JoinedDispatcher[_ET]:
82
+ raise NotImplementedError()
83
+
84
+ def __getattr__(self, name: str) -> _InstanceLevelDispatch[_ET]:
85
+ raise NotImplementedError()
86
+
87
+ @property
88
+ def _events(self) -> Type[_HasEventsDispatch[_ET]]:
89
+ raise NotImplementedError()
90
+
91
+
92
+ class _Dispatch(_DispatchCommon[_ET]):
93
+ """Mirror the event listening definitions of an Events class with
94
+ listener collections.
95
+
96
+ Classes which define a "dispatch" member will return a
97
+ non-instantiated :class:`._Dispatch` subclass when the member
98
+ is accessed at the class level. When the "dispatch" member is
99
+ accessed at the instance level of its owner, an instance
100
+ of the :class:`._Dispatch` class is returned.
101
+
102
+ A :class:`._Dispatch` class is generated for each :class:`.Events`
103
+ class defined, by the :meth:`._HasEventsDispatch._create_dispatcher_class`
104
+ method. The original :class:`.Events` classes remain untouched.
105
+ This decouples the construction of :class:`.Events` subclasses from
106
+ the implementation used by the event internals, and allows
107
+ inspecting tools like Sphinx to work in an unsurprising
108
+ way against the public API.
109
+
110
+ """
111
+
112
+ # "active_history" is an ORM case we add here. ideally a better
113
+ # system would be in place for ad-hoc attributes.
114
+ __slots__ = "_parent", "_instance_cls", "__dict__", "_empty_listeners"
115
+
116
+ _active_history: bool
117
+
118
+ _empty_listener_reg: MutableMapping[
119
+ Type[_ET], Dict[str, _EmptyListener[_ET]]
120
+ ] = weakref.WeakKeyDictionary()
121
+
122
+ _empty_listeners: Dict[str, _EmptyListener[_ET]]
123
+
124
+ _event_names: List[str]
125
+
126
+ _instance_cls: Optional[Type[_ET]]
127
+
128
+ _joined_dispatch_cls: Type[_JoinedDispatcher[_ET]]
129
+
130
+ _events: Type[_HasEventsDispatch[_ET]]
131
+ """reference back to the Events class.
132
+
133
+ Bidirectional against _HasEventsDispatch.dispatch
134
+
135
+ """
136
+
137
+ def __init__(
138
+ self,
139
+ parent: Optional[_Dispatch[_ET]],
140
+ instance_cls: Optional[Type[_ET]] = None,
141
+ ):
142
+ self._parent = parent
143
+ self._instance_cls = instance_cls
144
+
145
+ if instance_cls:
146
+ assert parent is not None
147
+ try:
148
+ self._empty_listeners = self._empty_listener_reg[instance_cls]
149
+ except KeyError:
150
+ self._empty_listeners = self._empty_listener_reg[
151
+ instance_cls
152
+ ] = {
153
+ ls.name: _EmptyListener(ls, instance_cls)
154
+ for ls in parent._event_descriptors
155
+ }
156
+ else:
157
+ self._empty_listeners = {}
158
+
159
+ def __getattr__(self, name: str) -> _InstanceLevelDispatch[_ET]:
160
+ # Assign EmptyListeners as attributes on demand
161
+ # to reduce startup time for new dispatch objects.
162
+ try:
163
+ ls = self._empty_listeners[name]
164
+ except KeyError:
165
+ raise AttributeError(name)
166
+ else:
167
+ setattr(self, ls.name, ls)
168
+ return ls
169
+
170
+ @property
171
+ def _event_descriptors(self) -> Iterator[_ClsLevelDispatch[_ET]]:
172
+ for k in self._event_names:
173
+ # Yield _ClsLevelDispatch related
174
+ # to relevant event name.
175
+ yield getattr(self, k)
176
+
177
+ def _listen(self, event_key: _EventKey[_ET], **kw: Any) -> None:
178
+ return self._events._listen(event_key, **kw)
179
+
180
+ def _for_class(self, instance_cls: Type[_ET]) -> _Dispatch[_ET]:
181
+ return self.__class__(self, instance_cls)
182
+
183
+ def _for_instance(self, instance: _ET) -> _Dispatch[_ET]:
184
+ instance_cls = instance.__class__
185
+ return self._for_class(instance_cls)
186
+
187
+ def _join(self, other: _DispatchCommon[_ET]) -> _JoinedDispatcher[_ET]:
188
+ """Create a 'join' of this :class:`._Dispatch` and another.
189
+
190
+ This new dispatcher will dispatch events to both
191
+ :class:`._Dispatch` objects.
192
+
193
+ """
194
+ assert "_joined_dispatch_cls" in self.__class__.__dict__
195
+
196
+ return self._joined_dispatch_cls(self, other)
197
+
198
+ def __reduce__(self) -> Union[str, Tuple[Any, ...]]:
199
+ return _UnpickleDispatch(), (self._instance_cls,)
200
+
201
+ def _update(
202
+ self, other: _Dispatch[_ET], only_propagate: bool = True
203
+ ) -> None:
204
+ """Populate from the listeners in another :class:`_Dispatch`
205
+ object."""
206
+ for ls in other._event_descriptors:
207
+ if isinstance(ls, _EmptyListener):
208
+ continue
209
+ getattr(self, ls.name).for_modify(self)._update(
210
+ ls, only_propagate=only_propagate
211
+ )
212
+
213
+ def _clear(self) -> None:
214
+ for ls in self._event_descriptors:
215
+ ls.for_modify(self).clear()
216
+
217
+
218
+ def _remove_dispatcher(cls: Type[_HasEventsDispatch[_ET]]) -> None:
219
+ for k in cls.dispatch._event_names:
220
+ _registrars[k].remove(cls)
221
+ if not _registrars[k]:
222
+ del _registrars[k]
223
+
224
+
225
+ class _HasEventsDispatch(Generic[_ET]):
226
+ _dispatch_target: Optional[Type[_ET]]
227
+ """class which will receive the .dispatch collection"""
228
+
229
+ dispatch: _Dispatch[_ET]
230
+ """reference back to the _Dispatch class.
231
+
232
+ Bidirectional against _Dispatch._events
233
+
234
+ """
235
+
236
+ if typing.TYPE_CHECKING:
237
+
238
+ def __getattr__(self, name: str) -> _InstanceLevelDispatch[_ET]: ...
239
+
240
+ def __init_subclass__(cls) -> None:
241
+ """Intercept new Event subclasses and create associated _Dispatch
242
+ classes."""
243
+
244
+ cls._create_dispatcher_class(cls.__name__, cls.__bases__, cls.__dict__)
245
+
246
+ @classmethod
247
+ def _accept_with(
248
+ cls, target: Union[_ET, Type[_ET]], identifier: str
249
+ ) -> Optional[Union[_ET, Type[_ET]]]:
250
+ raise NotImplementedError()
251
+
252
+ @classmethod
253
+ def _listen(
254
+ cls,
255
+ event_key: _EventKey[_ET],
256
+ *,
257
+ propagate: bool = False,
258
+ insert: bool = False,
259
+ named: bool = False,
260
+ asyncio: bool = False,
261
+ ) -> None:
262
+ raise NotImplementedError()
263
+
264
+ @staticmethod
265
+ def _set_dispatch(
266
+ klass: Type[_HasEventsDispatch[_ET]],
267
+ dispatch_cls: Type[_Dispatch[_ET]],
268
+ ) -> _Dispatch[_ET]:
269
+ # This allows an Events subclass to define additional utility
270
+ # methods made available to the target via
271
+ # "self.dispatch._events.<utilitymethod>"
272
+ # @staticmethod to allow easy "super" calls while in a metaclass
273
+ # constructor.
274
+ klass.dispatch = dispatch_cls(None)
275
+ dispatch_cls._events = klass
276
+ return klass.dispatch
277
+
278
+ @classmethod
279
+ def _create_dispatcher_class(
280
+ cls, classname: str, bases: Tuple[type, ...], dict_: Mapping[str, Any]
281
+ ) -> None:
282
+ """Create a :class:`._Dispatch` class corresponding to an
283
+ :class:`.Events` class."""
284
+
285
+ # there's all kinds of ways to do this,
286
+ # i.e. make a Dispatch class that shares the '_listen' method
287
+ # of the Event class, this is the straight monkeypatch.
288
+ if hasattr(cls, "dispatch"):
289
+ dispatch_base = cls.dispatch.__class__
290
+ else:
291
+ dispatch_base = _Dispatch
292
+
293
+ event_names = [k for k in dict_ if _is_event_name(k)]
294
+ dispatch_cls = cast(
295
+ "Type[_Dispatch[_ET]]",
296
+ type(
297
+ "%sDispatch" % classname,
298
+ (dispatch_base,),
299
+ {"__slots__": event_names},
300
+ ),
301
+ )
302
+
303
+ dispatch_cls._event_names = event_names
304
+ dispatch_inst = cls._set_dispatch(cls, dispatch_cls)
305
+ for k in dispatch_cls._event_names:
306
+ setattr(dispatch_inst, k, _ClsLevelDispatch(cls, dict_[k]))
307
+ _registrars[k].append(cls)
308
+
309
+ for super_ in dispatch_cls.__bases__:
310
+ if issubclass(super_, _Dispatch) and super_ is not _Dispatch:
311
+ for ls in super_._events.dispatch._event_descriptors:
312
+ setattr(dispatch_inst, ls.name, ls)
313
+ dispatch_cls._event_names.append(ls.name)
314
+
315
+ if getattr(cls, "_dispatch_target", None):
316
+ dispatch_target_cls = cls._dispatch_target
317
+ assert dispatch_target_cls is not None
318
+ if (
319
+ hasattr(dispatch_target_cls, "__slots__")
320
+ and "_slots_dispatch" in dispatch_target_cls.__slots__
321
+ ):
322
+ dispatch_target_cls.dispatch = slots_dispatcher(cls)
323
+ else:
324
+ dispatch_target_cls.dispatch = dispatcher(cls)
325
+
326
+ klass = type(
327
+ "Joined%s" % dispatch_cls.__name__,
328
+ (_JoinedDispatcher,),
329
+ {"__slots__": event_names},
330
+ )
331
+ dispatch_cls._joined_dispatch_cls = klass
332
+
333
+ # establish pickle capability by adding it to this module
334
+ globals()[klass.__name__] = klass
335
+
336
+
337
+ class _JoinedDispatcher(_DispatchCommon[_ET]):
338
+ """Represent a connection between two _Dispatch objects."""
339
+
340
+ __slots__ = "local", "parent", "_instance_cls"
341
+
342
+ local: _DispatchCommon[_ET]
343
+ parent: _DispatchCommon[_ET]
344
+ _instance_cls: Optional[Type[_ET]]
345
+
346
+ def __init__(
347
+ self, local: _DispatchCommon[_ET], parent: _DispatchCommon[_ET]
348
+ ):
349
+ self.local = local
350
+ self.parent = parent
351
+ self._instance_cls = self.local._instance_cls
352
+
353
+ def __reduce__(self) -> Any:
354
+ return (self.__class__, (self.local, self.parent))
355
+
356
+ def __getattr__(self, name: str) -> _JoinedListener[_ET]:
357
+ # Assign _JoinedListeners as attributes on demand
358
+ # to reduce startup time for new dispatch objects.
359
+ ls = getattr(self.local, name)
360
+ jl = _JoinedListener(self.parent, ls.name, ls)
361
+ setattr(self, ls.name, jl)
362
+ return jl
363
+
364
+ def _listen(self, event_key: _EventKey[_ET], **kw: Any) -> None:
365
+ return self.parent._listen(event_key, **kw)
366
+
367
+ @property
368
+ def _events(self) -> Type[_HasEventsDispatch[_ET]]:
369
+ return self.parent._events
370
+
371
+
372
+ class Events(_HasEventsDispatch[_ET]):
373
+ """Define event listening functions for a particular target type."""
374
+
375
+ @classmethod
376
+ def _accept_with(
377
+ cls, target: Union[_ET, Type[_ET]], identifier: str
378
+ ) -> Optional[Union[_ET, Type[_ET]]]:
379
+ def dispatch_is(*types: Type[Any]) -> bool:
380
+ return all(isinstance(target.dispatch, t) for t in types)
381
+
382
+ def dispatch_parent_is(t: Type[Any]) -> bool:
383
+ parent = cast("_JoinedDispatcher[_ET]", target.dispatch).parent
384
+ while isinstance(parent, _JoinedDispatcher):
385
+ parent = cast("_JoinedDispatcher[_ET]", parent).parent
386
+
387
+ return isinstance(parent, t)
388
+
389
+ # Mapper, ClassManager, Session override this to
390
+ # also accept classes, scoped_sessions, sessionmakers, etc.
391
+ if hasattr(target, "dispatch"):
392
+ if (
393
+ dispatch_is(cls.dispatch.__class__)
394
+ or dispatch_is(type, cls.dispatch.__class__)
395
+ or (
396
+ dispatch_is(_JoinedDispatcher)
397
+ and dispatch_parent_is(cls.dispatch.__class__)
398
+ )
399
+ ):
400
+ return target
401
+
402
+ return None
403
+
404
+ @classmethod
405
+ def _listen(
406
+ cls,
407
+ event_key: _EventKey[_ET],
408
+ *,
409
+ propagate: bool = False,
410
+ insert: bool = False,
411
+ named: bool = False,
412
+ asyncio: bool = False,
413
+ ) -> None:
414
+ event_key.base_listen(
415
+ propagate=propagate, insert=insert, named=named, asyncio=asyncio
416
+ )
417
+
418
+ @classmethod
419
+ def _remove(cls, event_key: _EventKey[_ET]) -> None:
420
+ event_key.remove()
421
+
422
+ @classmethod
423
+ def _clear(cls) -> None:
424
+ cls.dispatch._clear()
425
+
426
+
427
+ class dispatcher(Generic[_ET]):
428
+ """Descriptor used by target classes to
429
+ deliver the _Dispatch class at the class level
430
+ and produce new _Dispatch instances for target
431
+ instances.
432
+
433
+ """
434
+
435
+ def __init__(self, events: Type[_HasEventsDispatch[_ET]]):
436
+ self.dispatch = events.dispatch
437
+ self.events = events
438
+
439
+ @overload
440
+ def __get__(
441
+ self, obj: Literal[None], cls: Type[Any]
442
+ ) -> Type[_Dispatch[_ET]]: ...
443
+
444
+ @overload
445
+ def __get__(self, obj: Any, cls: Type[Any]) -> _DispatchCommon[_ET]: ...
446
+
447
+ def __get__(self, obj: Any, cls: Type[Any]) -> Any:
448
+ if obj is None:
449
+ return self.dispatch
450
+
451
+ disp = self.dispatch._for_instance(obj)
452
+ try:
453
+ obj.__dict__["dispatch"] = disp
454
+ except AttributeError as ae:
455
+ raise TypeError(
456
+ "target %r doesn't have __dict__, should it be "
457
+ "defining _slots_dispatch?" % (obj,)
458
+ ) from ae
459
+ return disp
460
+
461
+
462
+ class slots_dispatcher(dispatcher[_ET]):
463
+ def __get__(self, obj: Any, cls: Type[Any]) -> Any:
464
+ if obj is None:
465
+ return self.dispatch
466
+
467
+ if hasattr(obj, "_slots_dispatch"):
468
+ return obj._slots_dispatch
469
+
470
+ disp = self.dispatch._for_instance(obj)
471
+ obj._slots_dispatch = disp
472
+ return disp