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,450 @@
1
+ # ext/instrumentation.py
2
+ # Copyright (C) 2005-2026 the SQLAlchemy authors and contributors
3
+ # <see AUTHORS file>
4
+ #
5
+ # This module is part of SQLAlchemy and is released under
6
+ # the MIT License: https://www.opensource.org/licenses/mit-license.php
7
+ # mypy: ignore-errors
8
+
9
+ """Extensible class instrumentation.
10
+
11
+ The :mod:`sqlalchemy.ext.instrumentation` package provides for alternate
12
+ systems of class instrumentation within the ORM. Class instrumentation
13
+ refers to how the ORM places attributes on the class which maintain
14
+ data and track changes to that data, as well as event hooks installed
15
+ on the class.
16
+
17
+ .. note::
18
+ The extension package is provided for the benefit of integration
19
+ with other object management packages, which already perform
20
+ their own instrumentation. It is not intended for general use.
21
+
22
+ For examples of how the instrumentation extension is used,
23
+ see the example :ref:`examples_instrumentation`.
24
+
25
+ """
26
+ import weakref
27
+
28
+ from .. import util
29
+ from ..orm import attributes
30
+ from ..orm import base as orm_base
31
+ from ..orm import collections
32
+ from ..orm import exc as orm_exc
33
+ from ..orm import instrumentation as orm_instrumentation
34
+ from ..orm import util as orm_util
35
+ from ..orm.instrumentation import _default_dict_getter
36
+ from ..orm.instrumentation import _default_manager_getter
37
+ from ..orm.instrumentation import _default_opt_manager_getter
38
+ from ..orm.instrumentation import _default_state_getter
39
+ from ..orm.instrumentation import ClassManager
40
+ from ..orm.instrumentation import InstrumentationFactory
41
+
42
+
43
+ INSTRUMENTATION_MANAGER = "__sa_instrumentation_manager__"
44
+ """Attribute, elects custom instrumentation when present on a mapped class.
45
+
46
+ Allows a class to specify a slightly or wildly different technique for
47
+ tracking changes made to mapped attributes and collections.
48
+
49
+ Only one instrumentation implementation is allowed in a given object
50
+ inheritance hierarchy.
51
+
52
+ The value of this attribute must be a callable and will be passed a class
53
+ object. The callable must return one of:
54
+
55
+ - An instance of an :class:`.InstrumentationManager` or subclass
56
+ - An object implementing all or some of InstrumentationManager (TODO)
57
+ - A dictionary of callables, implementing all or some of the above (TODO)
58
+ - An instance of a :class:`.ClassManager` or subclass
59
+
60
+ This attribute is consulted by SQLAlchemy instrumentation
61
+ resolution, once the :mod:`sqlalchemy.ext.instrumentation` module
62
+ has been imported. If custom finders are installed in the global
63
+ instrumentation_finders list, they may or may not choose to honor this
64
+ attribute.
65
+
66
+ """
67
+
68
+
69
+ def find_native_user_instrumentation_hook(cls):
70
+ """Find user-specified instrumentation management for a class."""
71
+ return getattr(cls, INSTRUMENTATION_MANAGER, None)
72
+
73
+
74
+ instrumentation_finders = [find_native_user_instrumentation_hook]
75
+ """An extensible sequence of callables which return instrumentation
76
+ implementations
77
+
78
+ When a class is registered, each callable will be passed a class object.
79
+ If None is returned, the
80
+ next finder in the sequence is consulted. Otherwise the return must be an
81
+ instrumentation factory that follows the same guidelines as
82
+ sqlalchemy.ext.instrumentation.INSTRUMENTATION_MANAGER.
83
+
84
+ By default, the only finder is find_native_user_instrumentation_hook, which
85
+ searches for INSTRUMENTATION_MANAGER. If all finders return None, standard
86
+ ClassManager instrumentation is used.
87
+
88
+ """
89
+
90
+
91
+ class ExtendedInstrumentationRegistry(InstrumentationFactory):
92
+ """Extends :class:`.InstrumentationFactory` with additional
93
+ bookkeeping, to accommodate multiple types of
94
+ class managers.
95
+
96
+ """
97
+
98
+ _manager_finders = weakref.WeakKeyDictionary()
99
+ _state_finders = weakref.WeakKeyDictionary()
100
+ _dict_finders = weakref.WeakKeyDictionary()
101
+ _extended = False
102
+
103
+ def _locate_extended_factory(self, class_):
104
+ for finder in instrumentation_finders:
105
+ factory = finder(class_)
106
+ if factory is not None:
107
+ manager = self._extended_class_manager(class_, factory)
108
+ return manager, factory
109
+ else:
110
+ return None, None
111
+
112
+ def _check_conflicts(self, class_, factory):
113
+ existing_factories = self._collect_management_factories_for(
114
+ class_
115
+ ).difference([factory])
116
+ if existing_factories:
117
+ raise TypeError(
118
+ "multiple instrumentation implementations specified "
119
+ "in %s inheritance hierarchy: %r"
120
+ % (class_.__name__, list(existing_factories))
121
+ )
122
+
123
+ def _extended_class_manager(self, class_, factory):
124
+ manager = factory(class_)
125
+ if not isinstance(manager, ClassManager):
126
+ manager = _ClassInstrumentationAdapter(class_, manager)
127
+
128
+ if factory != ClassManager and not self._extended:
129
+ # somebody invoked a custom ClassManager.
130
+ # reinstall global "getter" functions with the more
131
+ # expensive ones.
132
+ self._extended = True
133
+ _install_instrumented_lookups()
134
+
135
+ self._manager_finders[class_] = manager.manager_getter()
136
+ self._state_finders[class_] = manager.state_getter()
137
+ self._dict_finders[class_] = manager.dict_getter()
138
+ return manager
139
+
140
+ def _collect_management_factories_for(self, cls):
141
+ """Return a collection of factories in play or specified for a
142
+ hierarchy.
143
+
144
+ Traverses the entire inheritance graph of a cls and returns a
145
+ collection of instrumentation factories for those classes. Factories
146
+ are extracted from active ClassManagers, if available, otherwise
147
+ instrumentation_finders is consulted.
148
+
149
+ """
150
+ hierarchy = util.class_hierarchy(cls)
151
+ factories = set()
152
+ for member in hierarchy:
153
+ manager = self.opt_manager_of_class(member)
154
+ if manager is not None:
155
+ factories.add(manager.factory)
156
+ else:
157
+ for finder in instrumentation_finders:
158
+ factory = finder(member)
159
+ if factory is not None:
160
+ break
161
+ else:
162
+ factory = None
163
+ factories.add(factory)
164
+ factories.discard(None)
165
+ return factories
166
+
167
+ def unregister(self, class_):
168
+ super().unregister(class_)
169
+ if class_ in self._manager_finders:
170
+ del self._manager_finders[class_]
171
+ del self._state_finders[class_]
172
+ del self._dict_finders[class_]
173
+
174
+ def opt_manager_of_class(self, cls):
175
+ try:
176
+ finder = self._manager_finders.get(
177
+ cls, _default_opt_manager_getter
178
+ )
179
+ except TypeError:
180
+ # due to weakref lookup on invalid object
181
+ return None
182
+ else:
183
+ return finder(cls)
184
+
185
+ def manager_of_class(self, cls):
186
+ try:
187
+ finder = self._manager_finders.get(cls, _default_manager_getter)
188
+ except TypeError:
189
+ # due to weakref lookup on invalid object
190
+ raise orm_exc.UnmappedClassError(
191
+ cls, f"Can't locate an instrumentation manager for class {cls}"
192
+ )
193
+ else:
194
+ manager = finder(cls)
195
+ if manager is None:
196
+ raise orm_exc.UnmappedClassError(
197
+ cls,
198
+ f"Can't locate an instrumentation manager for class {cls}",
199
+ )
200
+ return manager
201
+
202
+ def state_of(self, instance):
203
+ if instance is None:
204
+ raise AttributeError("None has no persistent state.")
205
+ return self._state_finders.get(
206
+ instance.__class__, _default_state_getter
207
+ )(instance)
208
+
209
+ def dict_of(self, instance):
210
+ if instance is None:
211
+ raise AttributeError("None has no persistent state.")
212
+ return self._dict_finders.get(
213
+ instance.__class__, _default_dict_getter
214
+ )(instance)
215
+
216
+
217
+ orm_instrumentation._instrumentation_factory = _instrumentation_factory = (
218
+ ExtendedInstrumentationRegistry()
219
+ )
220
+ orm_instrumentation.instrumentation_finders = instrumentation_finders
221
+
222
+
223
+ class InstrumentationManager:
224
+ """User-defined class instrumentation extension.
225
+
226
+ :class:`.InstrumentationManager` can be subclassed in order
227
+ to change
228
+ how class instrumentation proceeds. This class exists for
229
+ the purposes of integration with other object management
230
+ frameworks which would like to entirely modify the
231
+ instrumentation methodology of the ORM, and is not intended
232
+ for regular usage. For interception of class instrumentation
233
+ events, see :class:`.InstrumentationEvents`.
234
+
235
+ The API for this class should be considered as semi-stable,
236
+ and may change slightly with new releases.
237
+
238
+ """
239
+
240
+ # r4361 added a mandatory (cls) constructor to this interface.
241
+ # given that, perhaps class_ should be dropped from all of these
242
+ # signatures.
243
+
244
+ def __init__(self, class_):
245
+ pass
246
+
247
+ def manage(self, class_, manager):
248
+ setattr(class_, "_default_class_manager", manager)
249
+
250
+ def unregister(self, class_, manager):
251
+ delattr(class_, "_default_class_manager")
252
+
253
+ def manager_getter(self, class_):
254
+ def get(cls):
255
+ return cls._default_class_manager
256
+
257
+ return get
258
+
259
+ def instrument_attribute(self, class_, key, inst):
260
+ pass
261
+
262
+ def post_configure_attribute(self, class_, key, inst):
263
+ pass
264
+
265
+ def install_descriptor(self, class_, key, inst):
266
+ setattr(class_, key, inst)
267
+
268
+ def uninstall_descriptor(self, class_, key):
269
+ delattr(class_, key)
270
+
271
+ def install_member(self, class_, key, implementation):
272
+ setattr(class_, key, implementation)
273
+
274
+ def uninstall_member(self, class_, key):
275
+ delattr(class_, key)
276
+
277
+ def instrument_collection_class(self, class_, key, collection_class):
278
+ return collections.prepare_instrumentation(collection_class)
279
+
280
+ def get_instance_dict(self, class_, instance):
281
+ return instance.__dict__
282
+
283
+ def initialize_instance_dict(self, class_, instance):
284
+ pass
285
+
286
+ def install_state(self, class_, instance, state):
287
+ setattr(instance, "_default_state", state)
288
+
289
+ def remove_state(self, class_, instance):
290
+ delattr(instance, "_default_state")
291
+
292
+ def state_getter(self, class_):
293
+ return lambda instance: getattr(instance, "_default_state")
294
+
295
+ def dict_getter(self, class_):
296
+ return lambda inst: self.get_instance_dict(class_, inst)
297
+
298
+
299
+ class _ClassInstrumentationAdapter(ClassManager):
300
+ """Adapts a user-defined InstrumentationManager to a ClassManager."""
301
+
302
+ def __init__(self, class_, override):
303
+ self._adapted = override
304
+ self._get_state = self._adapted.state_getter(class_)
305
+ self._get_dict = self._adapted.dict_getter(class_)
306
+
307
+ ClassManager.__init__(self, class_)
308
+
309
+ def manage(self):
310
+ self._adapted.manage(self.class_, self)
311
+
312
+ def unregister(self):
313
+ self._adapted.unregister(self.class_, self)
314
+
315
+ def manager_getter(self):
316
+ return self._adapted.manager_getter(self.class_)
317
+
318
+ def instrument_attribute(self, key, inst, propagated=False):
319
+ ClassManager.instrument_attribute(self, key, inst, propagated)
320
+ if not propagated:
321
+ self._adapted.instrument_attribute(self.class_, key, inst)
322
+
323
+ def post_configure_attribute(self, key):
324
+ super().post_configure_attribute(key)
325
+ self._adapted.post_configure_attribute(self.class_, key, self[key])
326
+
327
+ def install_descriptor(self, key, inst):
328
+ self._adapted.install_descriptor(self.class_, key, inst)
329
+
330
+ def uninstall_descriptor(self, key):
331
+ self._adapted.uninstall_descriptor(self.class_, key)
332
+
333
+ def install_member(self, key, implementation):
334
+ self._adapted.install_member(self.class_, key, implementation)
335
+
336
+ def uninstall_member(self, key):
337
+ self._adapted.uninstall_member(self.class_, key)
338
+
339
+ def instrument_collection_class(self, key, collection_class):
340
+ return self._adapted.instrument_collection_class(
341
+ self.class_, key, collection_class
342
+ )
343
+
344
+ def initialize_collection(self, key, state, factory):
345
+ delegate = getattr(self._adapted, "initialize_collection", None)
346
+ if delegate:
347
+ return delegate(key, state, factory)
348
+ else:
349
+ return ClassManager.initialize_collection(
350
+ self, key, state, factory
351
+ )
352
+
353
+ def new_instance(self, state=None):
354
+ instance = self.class_.__new__(self.class_)
355
+ self.setup_instance(instance, state)
356
+ return instance
357
+
358
+ def _new_state_if_none(self, instance):
359
+ """Install a default InstanceState if none is present.
360
+
361
+ A private convenience method used by the __init__ decorator.
362
+ """
363
+ if self.has_state(instance):
364
+ return False
365
+ else:
366
+ return self.setup_instance(instance)
367
+
368
+ def setup_instance(self, instance, state=None):
369
+ self._adapted.initialize_instance_dict(self.class_, instance)
370
+
371
+ if state is None:
372
+ state = self._state_constructor(instance, self)
373
+
374
+ # the given instance is assumed to have no state
375
+ self._adapted.install_state(self.class_, instance, state)
376
+ return state
377
+
378
+ def teardown_instance(self, instance):
379
+ self._adapted.remove_state(self.class_, instance)
380
+
381
+ def has_state(self, instance):
382
+ try:
383
+ self._get_state(instance)
384
+ except orm_exc.NO_STATE:
385
+ return False
386
+ else:
387
+ return True
388
+
389
+ def state_getter(self):
390
+ return self._get_state
391
+
392
+ def dict_getter(self):
393
+ return self._get_dict
394
+
395
+
396
+ def _install_instrumented_lookups():
397
+ """Replace global class/object management functions
398
+ with ExtendedInstrumentationRegistry implementations, which
399
+ allow multiple types of class managers to be present,
400
+ at the cost of performance.
401
+
402
+ This function is called only by ExtendedInstrumentationRegistry
403
+ and unit tests specific to this behavior.
404
+
405
+ The _reinstall_default_lookups() function can be called
406
+ after this one to re-establish the default functions.
407
+
408
+ """
409
+ _install_lookups(
410
+ dict(
411
+ instance_state=_instrumentation_factory.state_of,
412
+ instance_dict=_instrumentation_factory.dict_of,
413
+ manager_of_class=_instrumentation_factory.manager_of_class,
414
+ opt_manager_of_class=_instrumentation_factory.opt_manager_of_class,
415
+ )
416
+ )
417
+
418
+
419
+ def _reinstall_default_lookups():
420
+ """Restore simplified lookups."""
421
+ _install_lookups(
422
+ dict(
423
+ instance_state=_default_state_getter,
424
+ instance_dict=_default_dict_getter,
425
+ manager_of_class=_default_manager_getter,
426
+ opt_manager_of_class=_default_opt_manager_getter,
427
+ )
428
+ )
429
+ _instrumentation_factory._extended = False
430
+
431
+
432
+ def _install_lookups(lookups):
433
+ global instance_state, instance_dict
434
+ global manager_of_class, opt_manager_of_class
435
+ instance_state = lookups["instance_state"]
436
+ instance_dict = lookups["instance_dict"]
437
+ manager_of_class = lookups["manager_of_class"]
438
+ opt_manager_of_class = lookups["opt_manager_of_class"]
439
+ orm_base.instance_state = attributes.instance_state = (
440
+ orm_instrumentation.instance_state
441
+ ) = instance_state
442
+ orm_base.instance_dict = attributes.instance_dict = (
443
+ orm_instrumentation.instance_dict
444
+ ) = instance_dict
445
+ orm_base.manager_of_class = attributes.manager_of_class = (
446
+ orm_instrumentation.manager_of_class
447
+ ) = manager_of_class
448
+ orm_base.opt_manager_of_class = orm_util.opt_manager_of_class = (
449
+ attributes.opt_manager_of_class
450
+ ) = orm_instrumentation.opt_manager_of_class = opt_manager_of_class