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.
- sqlalchemy/__init__.py +283 -0
- sqlalchemy/connectors/__init__.py +18 -0
- sqlalchemy/connectors/aioodbc.py +184 -0
- sqlalchemy/connectors/asyncio.py +429 -0
- sqlalchemy/connectors/pyodbc.py +250 -0
- sqlalchemy/cyextension/__init__.py +6 -0
- sqlalchemy/cyextension/collections.cp313t-win_amd64.pyd +0 -0
- sqlalchemy/cyextension/collections.pyx +409 -0
- sqlalchemy/cyextension/immutabledict.cp313t-win_amd64.pyd +0 -0
- sqlalchemy/cyextension/immutabledict.pxd +8 -0
- sqlalchemy/cyextension/immutabledict.pyx +133 -0
- sqlalchemy/cyextension/processors.cp313t-win_amd64.pyd +0 -0
- sqlalchemy/cyextension/processors.pyx +68 -0
- sqlalchemy/cyextension/resultproxy.cp313t-win_amd64.pyd +0 -0
- sqlalchemy/cyextension/resultproxy.pyx +102 -0
- sqlalchemy/cyextension/util.cp313t-win_amd64.pyd +0 -0
- sqlalchemy/cyextension/util.pyx +90 -0
- sqlalchemy/dialects/__init__.py +62 -0
- sqlalchemy/dialects/_typing.py +30 -0
- sqlalchemy/dialects/mssql/__init__.py +88 -0
- sqlalchemy/dialects/mssql/aioodbc.py +63 -0
- sqlalchemy/dialects/mssql/base.py +4093 -0
- sqlalchemy/dialects/mssql/information_schema.py +285 -0
- sqlalchemy/dialects/mssql/json.py +129 -0
- sqlalchemy/dialects/mssql/provision.py +185 -0
- sqlalchemy/dialects/mssql/pymssql.py +126 -0
- sqlalchemy/dialects/mssql/pyodbc.py +760 -0
- sqlalchemy/dialects/mysql/__init__.py +104 -0
- sqlalchemy/dialects/mysql/aiomysql.py +250 -0
- sqlalchemy/dialects/mysql/asyncmy.py +231 -0
- sqlalchemy/dialects/mysql/base.py +3949 -0
- sqlalchemy/dialects/mysql/cymysql.py +106 -0
- sqlalchemy/dialects/mysql/dml.py +225 -0
- sqlalchemy/dialects/mysql/enumerated.py +282 -0
- sqlalchemy/dialects/mysql/expression.py +146 -0
- sqlalchemy/dialects/mysql/json.py +91 -0
- sqlalchemy/dialects/mysql/mariadb.py +72 -0
- sqlalchemy/dialects/mysql/mariadbconnector.py +322 -0
- sqlalchemy/dialects/mysql/mysqlconnector.py +302 -0
- sqlalchemy/dialects/mysql/mysqldb.py +314 -0
- sqlalchemy/dialects/mysql/provision.py +153 -0
- sqlalchemy/dialects/mysql/pymysql.py +158 -0
- sqlalchemy/dialects/mysql/pyodbc.py +157 -0
- sqlalchemy/dialects/mysql/reflection.py +727 -0
- sqlalchemy/dialects/mysql/reserved_words.py +570 -0
- sqlalchemy/dialects/mysql/types.py +835 -0
- sqlalchemy/dialects/oracle/__init__.py +81 -0
- sqlalchemy/dialects/oracle/base.py +3802 -0
- sqlalchemy/dialects/oracle/cx_oracle.py +1555 -0
- sqlalchemy/dialects/oracle/dictionary.py +507 -0
- sqlalchemy/dialects/oracle/oracledb.py +941 -0
- sqlalchemy/dialects/oracle/provision.py +297 -0
- sqlalchemy/dialects/oracle/types.py +316 -0
- sqlalchemy/dialects/oracle/vector.py +365 -0
- sqlalchemy/dialects/postgresql/__init__.py +167 -0
- sqlalchemy/dialects/postgresql/_psycopg_common.py +189 -0
- sqlalchemy/dialects/postgresql/array.py +519 -0
- sqlalchemy/dialects/postgresql/asyncpg.py +1284 -0
- sqlalchemy/dialects/postgresql/base.py +5378 -0
- sqlalchemy/dialects/postgresql/dml.py +339 -0
- sqlalchemy/dialects/postgresql/ext.py +540 -0
- sqlalchemy/dialects/postgresql/hstore.py +406 -0
- sqlalchemy/dialects/postgresql/json.py +404 -0
- sqlalchemy/dialects/postgresql/named_types.py +524 -0
- sqlalchemy/dialects/postgresql/operators.py +129 -0
- sqlalchemy/dialects/postgresql/pg8000.py +669 -0
- sqlalchemy/dialects/postgresql/pg_catalog.py +326 -0
- sqlalchemy/dialects/postgresql/provision.py +183 -0
- sqlalchemy/dialects/postgresql/psycopg.py +862 -0
- sqlalchemy/dialects/postgresql/psycopg2.py +892 -0
- sqlalchemy/dialects/postgresql/psycopg2cffi.py +61 -0
- sqlalchemy/dialects/postgresql/ranges.py +1031 -0
- sqlalchemy/dialects/postgresql/types.py +313 -0
- sqlalchemy/dialects/sqlite/__init__.py +57 -0
- sqlalchemy/dialects/sqlite/aiosqlite.py +482 -0
- sqlalchemy/dialects/sqlite/base.py +3056 -0
- sqlalchemy/dialects/sqlite/dml.py +263 -0
- sqlalchemy/dialects/sqlite/json.py +92 -0
- sqlalchemy/dialects/sqlite/provision.py +229 -0
- sqlalchemy/dialects/sqlite/pysqlcipher.py +157 -0
- sqlalchemy/dialects/sqlite/pysqlite.py +756 -0
- sqlalchemy/dialects/type_migration_guidelines.txt +145 -0
- sqlalchemy/engine/__init__.py +62 -0
- sqlalchemy/engine/_py_processors.py +136 -0
- sqlalchemy/engine/_py_row.py +128 -0
- sqlalchemy/engine/_py_util.py +74 -0
- sqlalchemy/engine/base.py +3390 -0
- sqlalchemy/engine/characteristics.py +155 -0
- sqlalchemy/engine/create.py +893 -0
- sqlalchemy/engine/cursor.py +2298 -0
- sqlalchemy/engine/default.py +2394 -0
- sqlalchemy/engine/events.py +965 -0
- sqlalchemy/engine/interfaces.py +3471 -0
- sqlalchemy/engine/mock.py +134 -0
- sqlalchemy/engine/processors.py +61 -0
- sqlalchemy/engine/reflection.py +2102 -0
- sqlalchemy/engine/result.py +2399 -0
- sqlalchemy/engine/row.py +400 -0
- sqlalchemy/engine/strategies.py +16 -0
- sqlalchemy/engine/url.py +924 -0
- sqlalchemy/engine/util.py +167 -0
- sqlalchemy/event/__init__.py +26 -0
- sqlalchemy/event/api.py +220 -0
- sqlalchemy/event/attr.py +676 -0
- sqlalchemy/event/base.py +472 -0
- sqlalchemy/event/legacy.py +258 -0
- sqlalchemy/event/registry.py +390 -0
- sqlalchemy/events.py +17 -0
- sqlalchemy/exc.py +832 -0
- sqlalchemy/ext/__init__.py +11 -0
- sqlalchemy/ext/associationproxy.py +2027 -0
- sqlalchemy/ext/asyncio/__init__.py +25 -0
- sqlalchemy/ext/asyncio/base.py +281 -0
- sqlalchemy/ext/asyncio/engine.py +1471 -0
- sqlalchemy/ext/asyncio/exc.py +21 -0
- sqlalchemy/ext/asyncio/result.py +965 -0
- sqlalchemy/ext/asyncio/scoping.py +1599 -0
- sqlalchemy/ext/asyncio/session.py +1947 -0
- sqlalchemy/ext/automap.py +1701 -0
- sqlalchemy/ext/baked.py +570 -0
- sqlalchemy/ext/compiler.py +600 -0
- sqlalchemy/ext/declarative/__init__.py +65 -0
- sqlalchemy/ext/declarative/extensions.py +564 -0
- sqlalchemy/ext/horizontal_shard.py +478 -0
- sqlalchemy/ext/hybrid.py +1535 -0
- sqlalchemy/ext/indexable.py +364 -0
- sqlalchemy/ext/instrumentation.py +450 -0
- sqlalchemy/ext/mutable.py +1085 -0
- sqlalchemy/ext/mypy/__init__.py +6 -0
- sqlalchemy/ext/mypy/apply.py +324 -0
- sqlalchemy/ext/mypy/decl_class.py +515 -0
- sqlalchemy/ext/mypy/infer.py +590 -0
- sqlalchemy/ext/mypy/names.py +335 -0
- sqlalchemy/ext/mypy/plugin.py +303 -0
- sqlalchemy/ext/mypy/util.py +357 -0
- sqlalchemy/ext/orderinglist.py +439 -0
- sqlalchemy/ext/serializer.py +185 -0
- sqlalchemy/future/__init__.py +16 -0
- sqlalchemy/future/engine.py +15 -0
- sqlalchemy/inspection.py +174 -0
- sqlalchemy/log.py +288 -0
- sqlalchemy/orm/__init__.py +171 -0
- sqlalchemy/orm/_orm_constructors.py +2661 -0
- sqlalchemy/orm/_typing.py +179 -0
- sqlalchemy/orm/attributes.py +2845 -0
- sqlalchemy/orm/base.py +971 -0
- sqlalchemy/orm/bulk_persistence.py +2135 -0
- sqlalchemy/orm/clsregistry.py +571 -0
- sqlalchemy/orm/collections.py +1627 -0
- sqlalchemy/orm/context.py +3334 -0
- sqlalchemy/orm/decl_api.py +2004 -0
- sqlalchemy/orm/decl_base.py +2192 -0
- sqlalchemy/orm/dependency.py +1302 -0
- sqlalchemy/orm/descriptor_props.py +1092 -0
- sqlalchemy/orm/dynamic.py +300 -0
- sqlalchemy/orm/evaluator.py +379 -0
- sqlalchemy/orm/events.py +3252 -0
- sqlalchemy/orm/exc.py +237 -0
- sqlalchemy/orm/identity.py +302 -0
- sqlalchemy/orm/instrumentation.py +754 -0
- sqlalchemy/orm/interfaces.py +1496 -0
- sqlalchemy/orm/loading.py +1686 -0
- sqlalchemy/orm/mapped_collection.py +557 -0
- sqlalchemy/orm/mapper.py +4444 -0
- sqlalchemy/orm/path_registry.py +809 -0
- sqlalchemy/orm/persistence.py +1788 -0
- sqlalchemy/orm/properties.py +935 -0
- sqlalchemy/orm/query.py +3459 -0
- sqlalchemy/orm/relationships.py +3508 -0
- sqlalchemy/orm/scoping.py +2148 -0
- sqlalchemy/orm/session.py +5280 -0
- sqlalchemy/orm/state.py +1168 -0
- sqlalchemy/orm/state_changes.py +196 -0
- sqlalchemy/orm/strategies.py +3470 -0
- sqlalchemy/orm/strategy_options.py +2568 -0
- sqlalchemy/orm/sync.py +164 -0
- sqlalchemy/orm/unitofwork.py +796 -0
- sqlalchemy/orm/util.py +2403 -0
- sqlalchemy/orm/writeonly.py +674 -0
- sqlalchemy/pool/__init__.py +44 -0
- sqlalchemy/pool/base.py +1524 -0
- sqlalchemy/pool/events.py +375 -0
- sqlalchemy/pool/impl.py +588 -0
- sqlalchemy/py.typed +0 -0
- sqlalchemy/schema.py +69 -0
- sqlalchemy/sql/__init__.py +145 -0
- sqlalchemy/sql/_dml_constructors.py +132 -0
- sqlalchemy/sql/_elements_constructors.py +1872 -0
- sqlalchemy/sql/_orm_types.py +20 -0
- sqlalchemy/sql/_py_util.py +75 -0
- sqlalchemy/sql/_selectable_constructors.py +763 -0
- sqlalchemy/sql/_typing.py +482 -0
- sqlalchemy/sql/annotation.py +587 -0
- sqlalchemy/sql/base.py +2293 -0
- sqlalchemy/sql/cache_key.py +1057 -0
- sqlalchemy/sql/coercions.py +1404 -0
- sqlalchemy/sql/compiler.py +8081 -0
- sqlalchemy/sql/crud.py +1752 -0
- sqlalchemy/sql/ddl.py +1444 -0
- sqlalchemy/sql/default_comparator.py +551 -0
- sqlalchemy/sql/dml.py +1850 -0
- sqlalchemy/sql/elements.py +5589 -0
- sqlalchemy/sql/events.py +458 -0
- sqlalchemy/sql/expression.py +159 -0
- sqlalchemy/sql/functions.py +2158 -0
- sqlalchemy/sql/lambdas.py +1442 -0
- sqlalchemy/sql/naming.py +209 -0
- sqlalchemy/sql/operators.py +2623 -0
- sqlalchemy/sql/roles.py +323 -0
- sqlalchemy/sql/schema.py +6222 -0
- sqlalchemy/sql/selectable.py +7265 -0
- sqlalchemy/sql/sqltypes.py +3930 -0
- sqlalchemy/sql/traversals.py +1024 -0
- sqlalchemy/sql/type_api.py +2368 -0
- sqlalchemy/sql/util.py +1485 -0
- sqlalchemy/sql/visitors.py +1164 -0
- sqlalchemy/testing/__init__.py +96 -0
- sqlalchemy/testing/assertions.py +994 -0
- sqlalchemy/testing/assertsql.py +520 -0
- sqlalchemy/testing/asyncio.py +135 -0
- sqlalchemy/testing/config.py +434 -0
- sqlalchemy/testing/engines.py +483 -0
- sqlalchemy/testing/entities.py +117 -0
- sqlalchemy/testing/exclusions.py +476 -0
- sqlalchemy/testing/fixtures/__init__.py +28 -0
- sqlalchemy/testing/fixtures/base.py +384 -0
- sqlalchemy/testing/fixtures/mypy.py +332 -0
- sqlalchemy/testing/fixtures/orm.py +227 -0
- sqlalchemy/testing/fixtures/sql.py +482 -0
- sqlalchemy/testing/pickleable.py +155 -0
- sqlalchemy/testing/plugin/__init__.py +6 -0
- sqlalchemy/testing/plugin/bootstrap.py +51 -0
- sqlalchemy/testing/plugin/plugin_base.py +828 -0
- sqlalchemy/testing/plugin/pytestplugin.py +892 -0
- sqlalchemy/testing/profiling.py +329 -0
- sqlalchemy/testing/provision.py +603 -0
- sqlalchemy/testing/requirements.py +1945 -0
- sqlalchemy/testing/schema.py +198 -0
- sqlalchemy/testing/suite/__init__.py +19 -0
- sqlalchemy/testing/suite/test_cte.py +237 -0
- sqlalchemy/testing/suite/test_ddl.py +389 -0
- sqlalchemy/testing/suite/test_deprecations.py +153 -0
- sqlalchemy/testing/suite/test_dialect.py +776 -0
- sqlalchemy/testing/suite/test_insert.py +630 -0
- sqlalchemy/testing/suite/test_reflection.py +3557 -0
- sqlalchemy/testing/suite/test_results.py +504 -0
- sqlalchemy/testing/suite/test_rowcount.py +258 -0
- sqlalchemy/testing/suite/test_select.py +2010 -0
- sqlalchemy/testing/suite/test_sequence.py +317 -0
- sqlalchemy/testing/suite/test_types.py +2147 -0
- sqlalchemy/testing/suite/test_unicode_ddl.py +189 -0
- sqlalchemy/testing/suite/test_update_delete.py +139 -0
- sqlalchemy/testing/util.py +535 -0
- sqlalchemy/testing/warnings.py +52 -0
- sqlalchemy/types.py +74 -0
- sqlalchemy/util/__init__.py +162 -0
- sqlalchemy/util/_collections.py +712 -0
- sqlalchemy/util/_concurrency_py3k.py +288 -0
- sqlalchemy/util/_has_cy.py +40 -0
- sqlalchemy/util/_py_collections.py +541 -0
- sqlalchemy/util/compat.py +421 -0
- sqlalchemy/util/concurrency.py +110 -0
- sqlalchemy/util/deprecations.py +401 -0
- sqlalchemy/util/langhelpers.py +2203 -0
- sqlalchemy/util/preloaded.py +150 -0
- sqlalchemy/util/queue.py +322 -0
- sqlalchemy/util/tool_support.py +201 -0
- sqlalchemy/util/topological.py +120 -0
- sqlalchemy/util/typing.py +734 -0
- sqlalchemy-2.0.47.dist-info/METADATA +243 -0
- sqlalchemy-2.0.47.dist-info/RECORD +274 -0
- sqlalchemy-2.0.47.dist-info/WHEEL +5 -0
- sqlalchemy-2.0.47.dist-info/licenses/LICENSE +19 -0
- sqlalchemy-2.0.47.dist-info/top_level.txt +1 -0
sqlalchemy/event/base.py
ADDED
|
@@ -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
|