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.
- 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-win32.pyd +0 -0
- sqlalchemy/cyextension/collections.pyx +409 -0
- sqlalchemy/cyextension/immutabledict.cp313t-win32.pyd +0 -0
- sqlalchemy/cyextension/immutabledict.pxd +8 -0
- sqlalchemy/cyextension/immutabledict.pyx +133 -0
- sqlalchemy/cyextension/processors.cp313t-win32.pyd +0 -0
- sqlalchemy/cyextension/processors.pyx +68 -0
- sqlalchemy/cyextension/resultproxy.cp313t-win32.pyd +0 -0
- sqlalchemy/cyextension/resultproxy.pyx +102 -0
- sqlalchemy/cyextension/util.cp313t-win32.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
|
@@ -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()))
|
|
Binary file
|
|
@@ -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))
|
|
Binary file
|
|
@@ -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)
|
|
Binary file
|