SQLAlchemy 2.0.47__cp313-cp313t-win32.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (274) hide show
  1. sqlalchemy/__init__.py +283 -0
  2. sqlalchemy/connectors/__init__.py +18 -0
  3. sqlalchemy/connectors/aioodbc.py +184 -0
  4. sqlalchemy/connectors/asyncio.py +429 -0
  5. sqlalchemy/connectors/pyodbc.py +250 -0
  6. sqlalchemy/cyextension/__init__.py +6 -0
  7. sqlalchemy/cyextension/collections.cp313t-win32.pyd +0 -0
  8. sqlalchemy/cyextension/collections.pyx +409 -0
  9. sqlalchemy/cyextension/immutabledict.cp313t-win32.pyd +0 -0
  10. sqlalchemy/cyextension/immutabledict.pxd +8 -0
  11. sqlalchemy/cyextension/immutabledict.pyx +133 -0
  12. sqlalchemy/cyextension/processors.cp313t-win32.pyd +0 -0
  13. sqlalchemy/cyextension/processors.pyx +68 -0
  14. sqlalchemy/cyextension/resultproxy.cp313t-win32.pyd +0 -0
  15. sqlalchemy/cyextension/resultproxy.pyx +102 -0
  16. sqlalchemy/cyextension/util.cp313t-win32.pyd +0 -0
  17. sqlalchemy/cyextension/util.pyx +90 -0
  18. sqlalchemy/dialects/__init__.py +62 -0
  19. sqlalchemy/dialects/_typing.py +30 -0
  20. sqlalchemy/dialects/mssql/__init__.py +88 -0
  21. sqlalchemy/dialects/mssql/aioodbc.py +63 -0
  22. sqlalchemy/dialects/mssql/base.py +4093 -0
  23. sqlalchemy/dialects/mssql/information_schema.py +285 -0
  24. sqlalchemy/dialects/mssql/json.py +129 -0
  25. sqlalchemy/dialects/mssql/provision.py +185 -0
  26. sqlalchemy/dialects/mssql/pymssql.py +126 -0
  27. sqlalchemy/dialects/mssql/pyodbc.py +760 -0
  28. sqlalchemy/dialects/mysql/__init__.py +104 -0
  29. sqlalchemy/dialects/mysql/aiomysql.py +250 -0
  30. sqlalchemy/dialects/mysql/asyncmy.py +231 -0
  31. sqlalchemy/dialects/mysql/base.py +3949 -0
  32. sqlalchemy/dialects/mysql/cymysql.py +106 -0
  33. sqlalchemy/dialects/mysql/dml.py +225 -0
  34. sqlalchemy/dialects/mysql/enumerated.py +282 -0
  35. sqlalchemy/dialects/mysql/expression.py +146 -0
  36. sqlalchemy/dialects/mysql/json.py +91 -0
  37. sqlalchemy/dialects/mysql/mariadb.py +72 -0
  38. sqlalchemy/dialects/mysql/mariadbconnector.py +322 -0
  39. sqlalchemy/dialects/mysql/mysqlconnector.py +302 -0
  40. sqlalchemy/dialects/mysql/mysqldb.py +314 -0
  41. sqlalchemy/dialects/mysql/provision.py +153 -0
  42. sqlalchemy/dialects/mysql/pymysql.py +158 -0
  43. sqlalchemy/dialects/mysql/pyodbc.py +157 -0
  44. sqlalchemy/dialects/mysql/reflection.py +727 -0
  45. sqlalchemy/dialects/mysql/reserved_words.py +570 -0
  46. sqlalchemy/dialects/mysql/types.py +835 -0
  47. sqlalchemy/dialects/oracle/__init__.py +81 -0
  48. sqlalchemy/dialects/oracle/base.py +3802 -0
  49. sqlalchemy/dialects/oracle/cx_oracle.py +1555 -0
  50. sqlalchemy/dialects/oracle/dictionary.py +507 -0
  51. sqlalchemy/dialects/oracle/oracledb.py +941 -0
  52. sqlalchemy/dialects/oracle/provision.py +297 -0
  53. sqlalchemy/dialects/oracle/types.py +316 -0
  54. sqlalchemy/dialects/oracle/vector.py +365 -0
  55. sqlalchemy/dialects/postgresql/__init__.py +167 -0
  56. sqlalchemy/dialects/postgresql/_psycopg_common.py +189 -0
  57. sqlalchemy/dialects/postgresql/array.py +519 -0
  58. sqlalchemy/dialects/postgresql/asyncpg.py +1284 -0
  59. sqlalchemy/dialects/postgresql/base.py +5378 -0
  60. sqlalchemy/dialects/postgresql/dml.py +339 -0
  61. sqlalchemy/dialects/postgresql/ext.py +540 -0
  62. sqlalchemy/dialects/postgresql/hstore.py +406 -0
  63. sqlalchemy/dialects/postgresql/json.py +404 -0
  64. sqlalchemy/dialects/postgresql/named_types.py +524 -0
  65. sqlalchemy/dialects/postgresql/operators.py +129 -0
  66. sqlalchemy/dialects/postgresql/pg8000.py +669 -0
  67. sqlalchemy/dialects/postgresql/pg_catalog.py +326 -0
  68. sqlalchemy/dialects/postgresql/provision.py +183 -0
  69. sqlalchemy/dialects/postgresql/psycopg.py +862 -0
  70. sqlalchemy/dialects/postgresql/psycopg2.py +892 -0
  71. sqlalchemy/dialects/postgresql/psycopg2cffi.py +61 -0
  72. sqlalchemy/dialects/postgresql/ranges.py +1031 -0
  73. sqlalchemy/dialects/postgresql/types.py +313 -0
  74. sqlalchemy/dialects/sqlite/__init__.py +57 -0
  75. sqlalchemy/dialects/sqlite/aiosqlite.py +482 -0
  76. sqlalchemy/dialects/sqlite/base.py +3056 -0
  77. sqlalchemy/dialects/sqlite/dml.py +263 -0
  78. sqlalchemy/dialects/sqlite/json.py +92 -0
  79. sqlalchemy/dialects/sqlite/provision.py +229 -0
  80. sqlalchemy/dialects/sqlite/pysqlcipher.py +157 -0
  81. sqlalchemy/dialects/sqlite/pysqlite.py +756 -0
  82. sqlalchemy/dialects/type_migration_guidelines.txt +145 -0
  83. sqlalchemy/engine/__init__.py +62 -0
  84. sqlalchemy/engine/_py_processors.py +136 -0
  85. sqlalchemy/engine/_py_row.py +128 -0
  86. sqlalchemy/engine/_py_util.py +74 -0
  87. sqlalchemy/engine/base.py +3390 -0
  88. sqlalchemy/engine/characteristics.py +155 -0
  89. sqlalchemy/engine/create.py +893 -0
  90. sqlalchemy/engine/cursor.py +2298 -0
  91. sqlalchemy/engine/default.py +2394 -0
  92. sqlalchemy/engine/events.py +965 -0
  93. sqlalchemy/engine/interfaces.py +3471 -0
  94. sqlalchemy/engine/mock.py +134 -0
  95. sqlalchemy/engine/processors.py +61 -0
  96. sqlalchemy/engine/reflection.py +2102 -0
  97. sqlalchemy/engine/result.py +2399 -0
  98. sqlalchemy/engine/row.py +400 -0
  99. sqlalchemy/engine/strategies.py +16 -0
  100. sqlalchemy/engine/url.py +924 -0
  101. sqlalchemy/engine/util.py +167 -0
  102. sqlalchemy/event/__init__.py +26 -0
  103. sqlalchemy/event/api.py +220 -0
  104. sqlalchemy/event/attr.py +676 -0
  105. sqlalchemy/event/base.py +472 -0
  106. sqlalchemy/event/legacy.py +258 -0
  107. sqlalchemy/event/registry.py +390 -0
  108. sqlalchemy/events.py +17 -0
  109. sqlalchemy/exc.py +832 -0
  110. sqlalchemy/ext/__init__.py +11 -0
  111. sqlalchemy/ext/associationproxy.py +2027 -0
  112. sqlalchemy/ext/asyncio/__init__.py +25 -0
  113. sqlalchemy/ext/asyncio/base.py +281 -0
  114. sqlalchemy/ext/asyncio/engine.py +1471 -0
  115. sqlalchemy/ext/asyncio/exc.py +21 -0
  116. sqlalchemy/ext/asyncio/result.py +965 -0
  117. sqlalchemy/ext/asyncio/scoping.py +1599 -0
  118. sqlalchemy/ext/asyncio/session.py +1947 -0
  119. sqlalchemy/ext/automap.py +1701 -0
  120. sqlalchemy/ext/baked.py +570 -0
  121. sqlalchemy/ext/compiler.py +600 -0
  122. sqlalchemy/ext/declarative/__init__.py +65 -0
  123. sqlalchemy/ext/declarative/extensions.py +564 -0
  124. sqlalchemy/ext/horizontal_shard.py +478 -0
  125. sqlalchemy/ext/hybrid.py +1535 -0
  126. sqlalchemy/ext/indexable.py +364 -0
  127. sqlalchemy/ext/instrumentation.py +450 -0
  128. sqlalchemy/ext/mutable.py +1085 -0
  129. sqlalchemy/ext/mypy/__init__.py +6 -0
  130. sqlalchemy/ext/mypy/apply.py +324 -0
  131. sqlalchemy/ext/mypy/decl_class.py +515 -0
  132. sqlalchemy/ext/mypy/infer.py +590 -0
  133. sqlalchemy/ext/mypy/names.py +335 -0
  134. sqlalchemy/ext/mypy/plugin.py +303 -0
  135. sqlalchemy/ext/mypy/util.py +357 -0
  136. sqlalchemy/ext/orderinglist.py +439 -0
  137. sqlalchemy/ext/serializer.py +185 -0
  138. sqlalchemy/future/__init__.py +16 -0
  139. sqlalchemy/future/engine.py +15 -0
  140. sqlalchemy/inspection.py +174 -0
  141. sqlalchemy/log.py +288 -0
  142. sqlalchemy/orm/__init__.py +171 -0
  143. sqlalchemy/orm/_orm_constructors.py +2661 -0
  144. sqlalchemy/orm/_typing.py +179 -0
  145. sqlalchemy/orm/attributes.py +2845 -0
  146. sqlalchemy/orm/base.py +971 -0
  147. sqlalchemy/orm/bulk_persistence.py +2135 -0
  148. sqlalchemy/orm/clsregistry.py +571 -0
  149. sqlalchemy/orm/collections.py +1627 -0
  150. sqlalchemy/orm/context.py +3334 -0
  151. sqlalchemy/orm/decl_api.py +2004 -0
  152. sqlalchemy/orm/decl_base.py +2192 -0
  153. sqlalchemy/orm/dependency.py +1302 -0
  154. sqlalchemy/orm/descriptor_props.py +1092 -0
  155. sqlalchemy/orm/dynamic.py +300 -0
  156. sqlalchemy/orm/evaluator.py +379 -0
  157. sqlalchemy/orm/events.py +3252 -0
  158. sqlalchemy/orm/exc.py +237 -0
  159. sqlalchemy/orm/identity.py +302 -0
  160. sqlalchemy/orm/instrumentation.py +754 -0
  161. sqlalchemy/orm/interfaces.py +1496 -0
  162. sqlalchemy/orm/loading.py +1686 -0
  163. sqlalchemy/orm/mapped_collection.py +557 -0
  164. sqlalchemy/orm/mapper.py +4444 -0
  165. sqlalchemy/orm/path_registry.py +809 -0
  166. sqlalchemy/orm/persistence.py +1788 -0
  167. sqlalchemy/orm/properties.py +935 -0
  168. sqlalchemy/orm/query.py +3459 -0
  169. sqlalchemy/orm/relationships.py +3508 -0
  170. sqlalchemy/orm/scoping.py +2148 -0
  171. sqlalchemy/orm/session.py +5280 -0
  172. sqlalchemy/orm/state.py +1168 -0
  173. sqlalchemy/orm/state_changes.py +196 -0
  174. sqlalchemy/orm/strategies.py +3470 -0
  175. sqlalchemy/orm/strategy_options.py +2568 -0
  176. sqlalchemy/orm/sync.py +164 -0
  177. sqlalchemy/orm/unitofwork.py +796 -0
  178. sqlalchemy/orm/util.py +2403 -0
  179. sqlalchemy/orm/writeonly.py +674 -0
  180. sqlalchemy/pool/__init__.py +44 -0
  181. sqlalchemy/pool/base.py +1524 -0
  182. sqlalchemy/pool/events.py +375 -0
  183. sqlalchemy/pool/impl.py +588 -0
  184. sqlalchemy/py.typed +0 -0
  185. sqlalchemy/schema.py +69 -0
  186. sqlalchemy/sql/__init__.py +145 -0
  187. sqlalchemy/sql/_dml_constructors.py +132 -0
  188. sqlalchemy/sql/_elements_constructors.py +1872 -0
  189. sqlalchemy/sql/_orm_types.py +20 -0
  190. sqlalchemy/sql/_py_util.py +75 -0
  191. sqlalchemy/sql/_selectable_constructors.py +763 -0
  192. sqlalchemy/sql/_typing.py +482 -0
  193. sqlalchemy/sql/annotation.py +587 -0
  194. sqlalchemy/sql/base.py +2293 -0
  195. sqlalchemy/sql/cache_key.py +1057 -0
  196. sqlalchemy/sql/coercions.py +1404 -0
  197. sqlalchemy/sql/compiler.py +8081 -0
  198. sqlalchemy/sql/crud.py +1752 -0
  199. sqlalchemy/sql/ddl.py +1444 -0
  200. sqlalchemy/sql/default_comparator.py +551 -0
  201. sqlalchemy/sql/dml.py +1850 -0
  202. sqlalchemy/sql/elements.py +5589 -0
  203. sqlalchemy/sql/events.py +458 -0
  204. sqlalchemy/sql/expression.py +159 -0
  205. sqlalchemy/sql/functions.py +2158 -0
  206. sqlalchemy/sql/lambdas.py +1442 -0
  207. sqlalchemy/sql/naming.py +209 -0
  208. sqlalchemy/sql/operators.py +2623 -0
  209. sqlalchemy/sql/roles.py +323 -0
  210. sqlalchemy/sql/schema.py +6222 -0
  211. sqlalchemy/sql/selectable.py +7265 -0
  212. sqlalchemy/sql/sqltypes.py +3930 -0
  213. sqlalchemy/sql/traversals.py +1024 -0
  214. sqlalchemy/sql/type_api.py +2368 -0
  215. sqlalchemy/sql/util.py +1485 -0
  216. sqlalchemy/sql/visitors.py +1164 -0
  217. sqlalchemy/testing/__init__.py +96 -0
  218. sqlalchemy/testing/assertions.py +994 -0
  219. sqlalchemy/testing/assertsql.py +520 -0
  220. sqlalchemy/testing/asyncio.py +135 -0
  221. sqlalchemy/testing/config.py +434 -0
  222. sqlalchemy/testing/engines.py +483 -0
  223. sqlalchemy/testing/entities.py +117 -0
  224. sqlalchemy/testing/exclusions.py +476 -0
  225. sqlalchemy/testing/fixtures/__init__.py +28 -0
  226. sqlalchemy/testing/fixtures/base.py +384 -0
  227. sqlalchemy/testing/fixtures/mypy.py +332 -0
  228. sqlalchemy/testing/fixtures/orm.py +227 -0
  229. sqlalchemy/testing/fixtures/sql.py +482 -0
  230. sqlalchemy/testing/pickleable.py +155 -0
  231. sqlalchemy/testing/plugin/__init__.py +6 -0
  232. sqlalchemy/testing/plugin/bootstrap.py +51 -0
  233. sqlalchemy/testing/plugin/plugin_base.py +828 -0
  234. sqlalchemy/testing/plugin/pytestplugin.py +892 -0
  235. sqlalchemy/testing/profiling.py +329 -0
  236. sqlalchemy/testing/provision.py +603 -0
  237. sqlalchemy/testing/requirements.py +1945 -0
  238. sqlalchemy/testing/schema.py +198 -0
  239. sqlalchemy/testing/suite/__init__.py +19 -0
  240. sqlalchemy/testing/suite/test_cte.py +237 -0
  241. sqlalchemy/testing/suite/test_ddl.py +389 -0
  242. sqlalchemy/testing/suite/test_deprecations.py +153 -0
  243. sqlalchemy/testing/suite/test_dialect.py +776 -0
  244. sqlalchemy/testing/suite/test_insert.py +630 -0
  245. sqlalchemy/testing/suite/test_reflection.py +3557 -0
  246. sqlalchemy/testing/suite/test_results.py +504 -0
  247. sqlalchemy/testing/suite/test_rowcount.py +258 -0
  248. sqlalchemy/testing/suite/test_select.py +2010 -0
  249. sqlalchemy/testing/suite/test_sequence.py +317 -0
  250. sqlalchemy/testing/suite/test_types.py +2147 -0
  251. sqlalchemy/testing/suite/test_unicode_ddl.py +189 -0
  252. sqlalchemy/testing/suite/test_update_delete.py +139 -0
  253. sqlalchemy/testing/util.py +535 -0
  254. sqlalchemy/testing/warnings.py +52 -0
  255. sqlalchemy/types.py +74 -0
  256. sqlalchemy/util/__init__.py +162 -0
  257. sqlalchemy/util/_collections.py +712 -0
  258. sqlalchemy/util/_concurrency_py3k.py +288 -0
  259. sqlalchemy/util/_has_cy.py +40 -0
  260. sqlalchemy/util/_py_collections.py +541 -0
  261. sqlalchemy/util/compat.py +421 -0
  262. sqlalchemy/util/concurrency.py +110 -0
  263. sqlalchemy/util/deprecations.py +401 -0
  264. sqlalchemy/util/langhelpers.py +2203 -0
  265. sqlalchemy/util/preloaded.py +150 -0
  266. sqlalchemy/util/queue.py +322 -0
  267. sqlalchemy/util/tool_support.py +201 -0
  268. sqlalchemy/util/topological.py +120 -0
  269. sqlalchemy/util/typing.py +734 -0
  270. sqlalchemy-2.0.47.dist-info/METADATA +243 -0
  271. sqlalchemy-2.0.47.dist-info/RECORD +274 -0
  272. sqlalchemy-2.0.47.dist-info/WHEEL +5 -0
  273. sqlalchemy-2.0.47.dist-info/licenses/LICENSE +19 -0
  274. sqlalchemy-2.0.47.dist-info/top_level.txt +1 -0
@@ -0,0 +1,409 @@
1
+ # cyextension/collections.pyx
2
+ # Copyright (C) 2005-2024 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
+ cimport cython
8
+ from cpython.long cimport PyLong_FromLongLong
9
+ from cpython.set cimport PySet_Add
10
+
11
+ from collections.abc import Collection
12
+ from itertools import filterfalse
13
+
14
+ cdef bint add_not_present(set seen, object item, hashfunc):
15
+ hash_value = hashfunc(item)
16
+ if hash_value not in seen:
17
+ PySet_Add(seen, hash_value)
18
+ return True
19
+ else:
20
+ return False
21
+
22
+ cdef list cunique_list(seq, hashfunc=None):
23
+ cdef set seen = set()
24
+ if not hashfunc:
25
+ return [x for x in seq if x not in seen and not PySet_Add(seen, x)]
26
+ else:
27
+ return [x for x in seq if add_not_present(seen, x, hashfunc)]
28
+
29
+ def unique_list(seq, hashfunc=None):
30
+ return cunique_list(seq, hashfunc)
31
+
32
+ cdef class OrderedSet(set):
33
+
34
+ cdef list _list
35
+
36
+ @classmethod
37
+ def __class_getitem__(cls, key):
38
+ return cls
39
+
40
+ def __init__(self, d=None):
41
+ set.__init__(self)
42
+ if d is not None:
43
+ self._list = cunique_list(d)
44
+ set.update(self, self._list)
45
+ else:
46
+ self._list = []
47
+
48
+ cpdef OrderedSet copy(self):
49
+ cdef OrderedSet cp = OrderedSet.__new__(OrderedSet)
50
+ cp._list = list(self._list)
51
+ set.update(cp, cp._list)
52
+ return cp
53
+
54
+ @cython.final
55
+ cdef OrderedSet _from_list(self, list new_list):
56
+ cdef OrderedSet new = OrderedSet.__new__(OrderedSet)
57
+ new._list = new_list
58
+ set.update(new, new_list)
59
+ return new
60
+
61
+ def add(self, element):
62
+ if element not in self:
63
+ self._list.append(element)
64
+ PySet_Add(self, element)
65
+
66
+ def remove(self, element):
67
+ # set.remove will raise if element is not in self
68
+ set.remove(self, element)
69
+ self._list.remove(element)
70
+
71
+ def pop(self):
72
+ try:
73
+ value = self._list.pop()
74
+ except IndexError:
75
+ raise KeyError("pop from an empty set") from None
76
+ set.remove(self, value)
77
+ return value
78
+
79
+ def insert(self, Py_ssize_t pos, element):
80
+ if element not in self:
81
+ self._list.insert(pos, element)
82
+ PySet_Add(self, element)
83
+
84
+ def discard(self, element):
85
+ if element in self:
86
+ set.remove(self, element)
87
+ self._list.remove(element)
88
+
89
+ def clear(self):
90
+ set.clear(self)
91
+ self._list = []
92
+
93
+ def __getitem__(self, key):
94
+ return self._list[key]
95
+
96
+ def __iter__(self):
97
+ return iter(self._list)
98
+
99
+ def __add__(self, other):
100
+ return self.union(other)
101
+
102
+ def __repr__(self):
103
+ return "%s(%r)" % (self.__class__.__name__, self._list)
104
+
105
+ __str__ = __repr__
106
+
107
+ def update(self, *iterables):
108
+ for iterable in iterables:
109
+ for e in iterable:
110
+ if e not in self:
111
+ self._list.append(e)
112
+ set.add(self, e)
113
+
114
+ def __ior__(self, iterable):
115
+ self.update(iterable)
116
+ return self
117
+
118
+ def union(self, *other):
119
+ result = self.copy()
120
+ result.update(*other)
121
+ return result
122
+
123
+ def __or__(self, other):
124
+ return self.union(other)
125
+
126
+ def intersection(self, *other):
127
+ cdef set other_set = set.intersection(self, *other)
128
+ return self._from_list([a for a in self._list if a in other_set])
129
+
130
+ def __and__(self, other):
131
+ return self.intersection(other)
132
+
133
+ def symmetric_difference(self, other):
134
+ cdef set other_set
135
+ if isinstance(other, set):
136
+ other_set = <set> other
137
+ collection = other_set
138
+ elif isinstance(other, Collection):
139
+ collection = other
140
+ other_set = set(other)
141
+ else:
142
+ collection = list(other)
143
+ other_set = set(collection)
144
+ result = self._from_list([a for a in self._list if a not in other_set])
145
+ result.update(a for a in collection if a not in self)
146
+ return result
147
+
148
+ def __xor__(self, other):
149
+ return self.symmetric_difference(other)
150
+
151
+ def difference(self, *other):
152
+ cdef set other_set = set.difference(self, *other)
153
+ return self._from_list([a for a in self._list if a in other_set])
154
+
155
+ def __sub__(self, other):
156
+ return self.difference(other)
157
+
158
+ def intersection_update(self, *other):
159
+ set.intersection_update(self, *other)
160
+ self._list = [a for a in self._list if a in self]
161
+
162
+ def __iand__(self, other):
163
+ self.intersection_update(other)
164
+ return self
165
+
166
+ cpdef symmetric_difference_update(self, other):
167
+ collection = other if isinstance(other, Collection) else list(other)
168
+ set.symmetric_difference_update(self, collection)
169
+ self._list = [a for a in self._list if a in self]
170
+ self._list += [a for a in collection if a in self]
171
+
172
+ def __ixor__(self, other):
173
+ self.symmetric_difference_update(other)
174
+ return self
175
+
176
+ def difference_update(self, *other):
177
+ set.difference_update(self, *other)
178
+ self._list = [a for a in self._list if a in self]
179
+
180
+ def __isub__(self, other):
181
+ self.difference_update(other)
182
+ return self
183
+
184
+ cdef object cy_id(object item):
185
+ return PyLong_FromLongLong(<long long> (<void *>item))
186
+
187
+ # NOTE: cython 0.x will call __add__, __sub__, etc with the parameter swapped
188
+ # instead of the __rmeth__, so they need to check that also self is of the
189
+ # correct type. This is fixed in cython 3.x. See:
190
+ # https://docs.cython.org/en/latest/src/userguide/special_methods.html#arithmetic-methods
191
+ cdef class IdentitySet:
192
+ """A set that considers only object id() for uniqueness.
193
+
194
+ This strategy has edge cases for builtin types- it's possible to have
195
+ two 'foo' strings in one of these sets, for example. Use sparingly.
196
+
197
+ """
198
+
199
+ cdef dict _members
200
+
201
+ def __init__(self, iterable=None):
202
+ self._members = {}
203
+ if iterable:
204
+ self.update(iterable)
205
+
206
+ def add(self, value):
207
+ self._members[cy_id(value)] = value
208
+
209
+ def __contains__(self, value):
210
+ return cy_id(value) in self._members
211
+
212
+ cpdef remove(self, value):
213
+ del self._members[cy_id(value)]
214
+
215
+ def discard(self, value):
216
+ try:
217
+ self.remove(value)
218
+ except KeyError:
219
+ pass
220
+
221
+ def pop(self):
222
+ cdef tuple pair
223
+ try:
224
+ pair = self._members.popitem()
225
+ return pair[1]
226
+ except KeyError:
227
+ raise KeyError("pop from an empty set")
228
+
229
+ def clear(self):
230
+ self._members.clear()
231
+
232
+ def __eq__(self, other):
233
+ cdef IdentitySet other_
234
+ if isinstance(other, IdentitySet):
235
+ other_ = other
236
+ return self._members == other_._members
237
+ else:
238
+ return False
239
+
240
+ def __ne__(self, other):
241
+ cdef IdentitySet other_
242
+ if isinstance(other, IdentitySet):
243
+ other_ = other
244
+ return self._members != other_._members
245
+ else:
246
+ return True
247
+
248
+ cpdef issubset(self, iterable):
249
+ cdef IdentitySet other
250
+ if isinstance(iterable, self.__class__):
251
+ other = iterable
252
+ else:
253
+ other = self.__class__(iterable)
254
+
255
+ if len(self) > len(other):
256
+ return False
257
+ for m in filterfalse(other._members.__contains__, self._members):
258
+ return False
259
+ return True
260
+
261
+ def __le__(self, other):
262
+ if not isinstance(other, IdentitySet):
263
+ return NotImplemented
264
+ return self.issubset(other)
265
+
266
+ def __lt__(self, other):
267
+ if not isinstance(other, IdentitySet):
268
+ return NotImplemented
269
+ return len(self) < len(other) and self.issubset(other)
270
+
271
+ cpdef issuperset(self, iterable):
272
+ cdef IdentitySet other
273
+ if isinstance(iterable, self.__class__):
274
+ other = iterable
275
+ else:
276
+ other = self.__class__(iterable)
277
+
278
+ if len(self) < len(other):
279
+ return False
280
+ for m in filterfalse(self._members.__contains__, other._members):
281
+ return False
282
+ return True
283
+
284
+ def __ge__(self, other):
285
+ if not isinstance(other, IdentitySet):
286
+ return NotImplemented
287
+ return self.issuperset(other)
288
+
289
+ def __gt__(self, other):
290
+ if not isinstance(other, IdentitySet):
291
+ return NotImplemented
292
+ return len(self) > len(other) and self.issuperset(other)
293
+
294
+ cpdef IdentitySet union(self, iterable):
295
+ cdef IdentitySet result = self.__class__()
296
+ result._members.update(self._members)
297
+ result.update(iterable)
298
+ return result
299
+
300
+ def __or__(self, other):
301
+ if not isinstance(other, IdentitySet) or not isinstance(self, IdentitySet):
302
+ return NotImplemented
303
+ return self.union(other)
304
+
305
+ cpdef update(self, iterable):
306
+ for obj in iterable:
307
+ self._members[cy_id(obj)] = obj
308
+
309
+ def __ior__(self, other):
310
+ if not isinstance(other, IdentitySet):
311
+ return NotImplemented
312
+ self.update(other)
313
+ return self
314
+
315
+ cpdef IdentitySet difference(self, iterable):
316
+ cdef IdentitySet result = self.__new__(self.__class__)
317
+ if isinstance(iterable, self.__class__):
318
+ other = (<IdentitySet>iterable)._members
319
+ else:
320
+ other = {cy_id(obj) for obj in iterable}
321
+ result._members = {k:v for k, v in self._members.items() if k not in other}
322
+ return result
323
+
324
+ def __sub__(self, other):
325
+ if not isinstance(other, IdentitySet) or not isinstance(self, IdentitySet):
326
+ return NotImplemented
327
+ return self.difference(other)
328
+
329
+ cpdef difference_update(self, iterable):
330
+ cdef IdentitySet other = self.difference(iterable)
331
+ self._members = other._members
332
+
333
+ def __isub__(self, other):
334
+ if not isinstance(other, IdentitySet):
335
+ return NotImplemented
336
+ self.difference_update(other)
337
+ return self
338
+
339
+ cpdef IdentitySet intersection(self, iterable):
340
+ cdef IdentitySet result = self.__new__(self.__class__)
341
+ if isinstance(iterable, self.__class__):
342
+ other = (<IdentitySet>iterable)._members
343
+ else:
344
+ other = {cy_id(obj) for obj in iterable}
345
+ result._members = {k: v for k, v in self._members.items() if k in other}
346
+ return result
347
+
348
+ def __and__(self, other):
349
+ if not isinstance(other, IdentitySet) or not isinstance(self, IdentitySet):
350
+ return NotImplemented
351
+ return self.intersection(other)
352
+
353
+ cpdef intersection_update(self, iterable):
354
+ cdef IdentitySet other = self.intersection(iterable)
355
+ self._members = other._members
356
+
357
+ def __iand__(self, other):
358
+ if not isinstance(other, IdentitySet):
359
+ return NotImplemented
360
+ self.intersection_update(other)
361
+ return self
362
+
363
+ cpdef IdentitySet symmetric_difference(self, iterable):
364
+ cdef IdentitySet result = self.__new__(self.__class__)
365
+ cdef dict other
366
+ if isinstance(iterable, self.__class__):
367
+ other = (<IdentitySet>iterable)._members
368
+ else:
369
+ other = {cy_id(obj): obj for obj in iterable}
370
+ result._members = {k: v for k, v in self._members.items() if k not in other}
371
+ result._members.update(
372
+ [(k, v) for k, v in other.items() if k not in self._members]
373
+ )
374
+ return result
375
+
376
+ def __xor__(self, other):
377
+ if not isinstance(other, IdentitySet) or not isinstance(self, IdentitySet):
378
+ return NotImplemented
379
+ return self.symmetric_difference(other)
380
+
381
+ cpdef symmetric_difference_update(self, iterable):
382
+ cdef IdentitySet other = self.symmetric_difference(iterable)
383
+ self._members = other._members
384
+
385
+ def __ixor__(self, other):
386
+ if not isinstance(other, IdentitySet):
387
+ return NotImplemented
388
+ self.symmetric_difference(other)
389
+ return self
390
+
391
+ cpdef IdentitySet copy(self):
392
+ cdef IdentitySet cp = self.__new__(self.__class__)
393
+ cp._members = self._members.copy()
394
+ return cp
395
+
396
+ def __copy__(self):
397
+ return self.copy()
398
+
399
+ def __len__(self):
400
+ return len(self._members)
401
+
402
+ def __iter__(self):
403
+ return iter(self._members.values())
404
+
405
+ def __hash__(self):
406
+ raise TypeError("set objects are unhashable")
407
+
408
+ def __repr__(self):
409
+ return "%s(%r)" % (type(self).__name__, list(self._members.values()))
@@ -0,0 +1,8 @@
1
+ # cyextension/immutabledict.pxd
2
+ # Copyright (C) 2005-2024 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
+ cdef class immutabledict(dict):
8
+ pass
@@ -0,0 +1,133 @@
1
+ # cyextension/immutabledict.pyx
2
+ # Copyright (C) 2005-2024 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
+ from cpython.dict cimport PyDict_New, PyDict_Update, PyDict_Size
8
+
9
+
10
+ def _readonly_fn(obj):
11
+ raise TypeError(
12
+ "%s object is immutable and/or readonly" % obj.__class__.__name__)
13
+
14
+
15
+ def _immutable_fn(obj):
16
+ raise TypeError(
17
+ "%s object is immutable" % obj.__class__.__name__)
18
+
19
+
20
+ class ReadOnlyContainer:
21
+
22
+ __slots__ = ()
23
+
24
+ def _readonly(self, *a,**kw):
25
+ _readonly_fn(self)
26
+
27
+ __delitem__ = __setitem__ = __setattr__ = _readonly
28
+
29
+
30
+ class ImmutableDictBase(dict):
31
+ def _immutable(self, *a,**kw):
32
+ _immutable_fn(self)
33
+
34
+ @classmethod
35
+ def __class_getitem__(cls, key):
36
+ return cls
37
+
38
+ __delitem__ = __setitem__ = __setattr__ = _immutable
39
+ clear = pop = popitem = setdefault = update = _immutable
40
+
41
+
42
+ cdef class immutabledict(dict):
43
+ def __repr__(self):
44
+ return f"immutabledict({dict.__repr__(self)})"
45
+
46
+ @classmethod
47
+ def __class_getitem__(cls, key):
48
+ return cls
49
+
50
+ def union(self, *args, **kw):
51
+ cdef dict to_merge = None
52
+ cdef immutabledict result
53
+ cdef Py_ssize_t args_len = len(args)
54
+ if args_len > 1:
55
+ raise TypeError(
56
+ f'union expected at most 1 argument, got {args_len}'
57
+ )
58
+ if args_len == 1:
59
+ attribute = args[0]
60
+ if isinstance(attribute, dict):
61
+ to_merge = <dict> attribute
62
+ if to_merge is None:
63
+ to_merge = dict(*args, **kw)
64
+
65
+ if PyDict_Size(to_merge) == 0:
66
+ return self
67
+
68
+ # new + update is faster than immutabledict(self)
69
+ result = immutabledict()
70
+ PyDict_Update(result, self)
71
+ PyDict_Update(result, to_merge)
72
+ return result
73
+
74
+ def merge_with(self, *other):
75
+ cdef immutabledict result = None
76
+ cdef object d
77
+ cdef bint update = False
78
+ if not other:
79
+ return self
80
+ for d in other:
81
+ if d:
82
+ if update == False:
83
+ update = True
84
+ # new + update is faster than immutabledict(self)
85
+ result = immutabledict()
86
+ PyDict_Update(result, self)
87
+ PyDict_Update(
88
+ result, <dict>(d if isinstance(d, dict) else dict(d))
89
+ )
90
+
91
+ return self if update == False else result
92
+
93
+ def copy(self):
94
+ return self
95
+
96
+ def __reduce__(self):
97
+ return immutabledict, (dict(self), )
98
+
99
+ def __delitem__(self, k):
100
+ _immutable_fn(self)
101
+
102
+ def __setitem__(self, k, v):
103
+ _immutable_fn(self)
104
+
105
+ def __setattr__(self, k, v):
106
+ _immutable_fn(self)
107
+
108
+ def clear(self, *args, **kw):
109
+ _immutable_fn(self)
110
+
111
+ def pop(self, *args, **kw):
112
+ _immutable_fn(self)
113
+
114
+ def popitem(self, *args, **kw):
115
+ _immutable_fn(self)
116
+
117
+ def setdefault(self, *args, **kw):
118
+ _immutable_fn(self)
119
+
120
+ def update(self, *args, **kw):
121
+ _immutable_fn(self)
122
+
123
+ # PEP 584
124
+ def __ior__(self, other):
125
+ _immutable_fn(self)
126
+
127
+ def __or__(self, other):
128
+ return immutabledict(dict.__or__(self, other))
129
+
130
+ def __ror__(self, other):
131
+ # NOTE: this is used only in cython 3.x;
132
+ # version 0.x will call __or__ with args inversed
133
+ return immutabledict(dict.__ror__(self, other))
@@ -0,0 +1,68 @@
1
+ # cyextension/processors.pyx
2
+ # Copyright (C) 2005-2024 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
+ import datetime
8
+ from datetime import datetime as datetime_cls
9
+ from datetime import time as time_cls
10
+ from datetime import date as date_cls
11
+ import re
12
+
13
+ from cpython.object cimport PyObject_Str
14
+ from cpython.unicode cimport PyUnicode_AsASCIIString, PyUnicode_Check, PyUnicode_Decode
15
+ from libc.stdio cimport sscanf
16
+
17
+
18
+ def int_to_boolean(value):
19
+ if value is None:
20
+ return None
21
+ return True if value else False
22
+
23
+ def to_str(value):
24
+ return PyObject_Str(value) if value is not None else None
25
+
26
+ def to_float(value):
27
+ return float(value) if value is not None else None
28
+
29
+ cdef inline bytes to_bytes(object value, str type_name):
30
+ try:
31
+ return PyUnicode_AsASCIIString(value)
32
+ except Exception as e:
33
+ raise ValueError(
34
+ f"Couldn't parse {type_name} string '{value!r}' "
35
+ "- value is not a string."
36
+ ) from e
37
+
38
+ def str_to_datetime(value):
39
+ if value is not None:
40
+ value = datetime_cls.fromisoformat(value)
41
+ return value
42
+
43
+ def str_to_time(value):
44
+ if value is not None:
45
+ value = time_cls.fromisoformat(value)
46
+ return value
47
+
48
+
49
+ def str_to_date(value):
50
+ if value is not None:
51
+ value = date_cls.fromisoformat(value)
52
+ return value
53
+
54
+
55
+
56
+ cdef class DecimalResultProcessor:
57
+ cdef object type_
58
+ cdef str format_
59
+
60
+ def __cinit__(self, type_, format_):
61
+ self.type_ = type_
62
+ self.format_ = format_
63
+
64
+ def process(self, object value):
65
+ if value is None:
66
+ return None
67
+ else:
68
+ return self.type_(self.format_ % value)