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,892 @@
1
+ # dialects/postgresql/psycopg2.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
+ r"""
10
+ .. dialect:: postgresql+psycopg2
11
+ :name: psycopg2
12
+ :dbapi: psycopg2
13
+ :connectstring: postgresql+psycopg2://user:password@host:port/dbname[?key=value&key=value...]
14
+ :url: https://pypi.org/project/psycopg2/
15
+
16
+ .. _psycopg2_toplevel:
17
+
18
+ psycopg2 Connect Arguments
19
+ --------------------------
20
+
21
+ Keyword arguments that are specific to the SQLAlchemy psycopg2 dialect
22
+ may be passed to :func:`_sa.create_engine()`, and include the following:
23
+
24
+
25
+ * ``isolation_level``: This option, available for all PostgreSQL dialects,
26
+ includes the ``AUTOCOMMIT`` isolation level when using the psycopg2
27
+ dialect. This option sets the **default** isolation level for the
28
+ connection that is set immediately upon connection to the database before
29
+ the connection is pooled. This option is generally superseded by the more
30
+ modern :paramref:`_engine.Connection.execution_options.isolation_level`
31
+ execution option, detailed at :ref:`dbapi_autocommit`.
32
+
33
+ .. seealso::
34
+
35
+ :ref:`psycopg2_isolation_level`
36
+
37
+ :ref:`dbapi_autocommit`
38
+
39
+
40
+ * ``client_encoding``: sets the client encoding in a libpq-agnostic way,
41
+ using psycopg2's ``set_client_encoding()`` method.
42
+
43
+ .. seealso::
44
+
45
+ :ref:`psycopg2_unicode`
46
+
47
+
48
+ * ``executemany_mode``, ``executemany_batch_page_size``,
49
+ ``executemany_values_page_size``: Allows use of psycopg2
50
+ extensions for optimizing "executemany"-style queries. See the referenced
51
+ section below for details.
52
+
53
+ .. seealso::
54
+
55
+ :ref:`psycopg2_executemany_mode`
56
+
57
+ .. tip::
58
+
59
+ The above keyword arguments are **dialect** keyword arguments, meaning
60
+ that they are passed as explicit keyword arguments to :func:`_sa.create_engine()`::
61
+
62
+ engine = create_engine(
63
+ "postgresql+psycopg2://scott:tiger@localhost/test",
64
+ isolation_level="SERIALIZABLE",
65
+ )
66
+
67
+ These should not be confused with **DBAPI** connect arguments, which
68
+ are passed as part of the :paramref:`_sa.create_engine.connect_args`
69
+ dictionary and/or are passed in the URL query string, as detailed in
70
+ the section :ref:`custom_dbapi_args`.
71
+
72
+ .. _psycopg2_ssl:
73
+
74
+ SSL Connections
75
+ ---------------
76
+
77
+ The psycopg2 module has a connection argument named ``sslmode`` for
78
+ controlling its behavior regarding secure (SSL) connections. The default is
79
+ ``sslmode=prefer``; it will attempt an SSL connection and if that fails it
80
+ will fall back to an unencrypted connection. ``sslmode=require`` may be used
81
+ to ensure that only secure connections are established. Consult the
82
+ psycopg2 / libpq documentation for further options that are available.
83
+
84
+ Note that ``sslmode`` is specific to psycopg2 so it is included in the
85
+ connection URI::
86
+
87
+ engine = sa.create_engine(
88
+ "postgresql+psycopg2://scott:tiger@192.168.0.199:5432/test?sslmode=require"
89
+ )
90
+
91
+ Unix Domain Connections
92
+ ------------------------
93
+
94
+ psycopg2 supports connecting via Unix domain connections. When the ``host``
95
+ portion of the URL is omitted, SQLAlchemy passes ``None`` to psycopg2,
96
+ which specifies Unix-domain communication rather than TCP/IP communication::
97
+
98
+ create_engine("postgresql+psycopg2://user:password@/dbname")
99
+
100
+ By default, the socket file used is to connect to a Unix-domain socket
101
+ in ``/tmp``, or whatever socket directory was specified when PostgreSQL
102
+ was built. This value can be overridden by passing a pathname to psycopg2,
103
+ using ``host`` as an additional keyword argument::
104
+
105
+ create_engine(
106
+ "postgresql+psycopg2://user:password@/dbname?host=/var/lib/postgresql"
107
+ )
108
+
109
+ .. warning:: The format accepted here allows for a hostname in the main URL
110
+ in addition to the "host" query string argument. **When using this URL
111
+ format, the initial host is silently ignored**. That is, this URL::
112
+
113
+ engine = create_engine(
114
+ "postgresql+psycopg2://user:password@myhost1/dbname?host=myhost2"
115
+ )
116
+
117
+ Above, the hostname ``myhost1`` is **silently ignored and discarded.** The
118
+ host which is connected is the ``myhost2`` host.
119
+
120
+ This is to maintain some degree of compatibility with PostgreSQL's own URL
121
+ format which has been tested to behave the same way and for which tools like
122
+ PifPaf hardcode two hostnames.
123
+
124
+ .. seealso::
125
+
126
+ `PQconnectdbParams \
127
+ <https://www.postgresql.org/docs/current/static/libpq-connect.html#LIBPQ-PQCONNECTDBPARAMS>`_
128
+
129
+ .. _psycopg2_multi_host:
130
+
131
+ Specifying multiple fallback hosts
132
+ -----------------------------------
133
+
134
+ psycopg2 supports multiple connection points in the connection string.
135
+ When the ``host`` parameter is used multiple times in the query section of
136
+ the URL, SQLAlchemy will create a single string of the host and port
137
+ information provided to make the connections. Tokens may consist of
138
+ ``host::port`` or just ``host``; in the latter case, the default port
139
+ is selected by libpq. In the example below, three host connections
140
+ are specified, for ``HostA::PortA``, ``HostB`` connecting to the default port,
141
+ and ``HostC::PortC``::
142
+
143
+ create_engine(
144
+ "postgresql+psycopg2://user:password@/dbname?host=HostA:PortA&host=HostB&host=HostC:PortC"
145
+ )
146
+
147
+ As an alternative, libpq query string format also may be used; this specifies
148
+ ``host`` and ``port`` as single query string arguments with comma-separated
149
+ lists - the default port can be chosen by indicating an empty value
150
+ in the comma separated list::
151
+
152
+ create_engine(
153
+ "postgresql+psycopg2://user:password@/dbname?host=HostA,HostB,HostC&port=PortA,,PortC"
154
+ )
155
+
156
+ With either URL style, connections to each host is attempted based on a
157
+ configurable strategy, which may be configured using the libpq
158
+ ``target_session_attrs`` parameter. Per libpq this defaults to ``any``
159
+ which indicates a connection to each host is then attempted until a connection is successful.
160
+ Other strategies include ``primary``, ``prefer-standby``, etc. The complete
161
+ list is documented by PostgreSQL at
162
+ `libpq connection strings <https://www.postgresql.org/docs/current/libpq-connect.html#LIBPQ-CONNSTRING>`_.
163
+
164
+ For example, to indicate two hosts using the ``primary`` strategy::
165
+
166
+ create_engine(
167
+ "postgresql+psycopg2://user:password@/dbname?host=HostA:PortA&host=HostB&host=HostC:PortC&target_session_attrs=primary"
168
+ )
169
+
170
+ .. versionchanged:: 1.4.40 Port specification in psycopg2 multiple host format
171
+ is repaired, previously ports were not correctly interpreted in this context.
172
+ libpq comma-separated format is also now supported.
173
+
174
+ .. versionadded:: 1.3.20 Support for multiple hosts in PostgreSQL connection
175
+ string.
176
+
177
+ .. seealso::
178
+
179
+ `libpq connection strings <https://www.postgresql.org/docs/current/libpq-connect.html#LIBPQ-CONNSTRING>`_ - please refer
180
+ to this section in the libpq documentation for complete background on multiple host support.
181
+
182
+
183
+ Empty DSN Connections / Environment Variable Connections
184
+ ---------------------------------------------------------
185
+
186
+ The psycopg2 DBAPI can connect to PostgreSQL by passing an empty DSN to the
187
+ libpq client library, which by default indicates to connect to a localhost
188
+ PostgreSQL database that is open for "trust" connections. This behavior can be
189
+ further tailored using a particular set of environment variables which are
190
+ prefixed with ``PG_...``, which are consumed by ``libpq`` to take the place of
191
+ any or all elements of the connection string.
192
+
193
+ For this form, the URL can be passed without any elements other than the
194
+ initial scheme::
195
+
196
+ engine = create_engine("postgresql+psycopg2://")
197
+
198
+ In the above form, a blank "dsn" string is passed to the ``psycopg2.connect()``
199
+ function which in turn represents an empty DSN passed to libpq.
200
+
201
+ .. versionadded:: 1.3.2 support for parameter-less connections with psycopg2.
202
+
203
+ .. seealso::
204
+
205
+ `Environment Variables\
206
+ <https://www.postgresql.org/docs/current/libpq-envars.html>`_ -
207
+ PostgreSQL documentation on how to use ``PG_...``
208
+ environment variables for connections.
209
+
210
+ .. _psycopg2_execution_options:
211
+
212
+ Per-Statement/Connection Execution Options
213
+ -------------------------------------------
214
+
215
+ The following DBAPI-specific options are respected when used with
216
+ :meth:`_engine.Connection.execution_options`,
217
+ :meth:`.Executable.execution_options`,
218
+ :meth:`_query.Query.execution_options`,
219
+ in addition to those not specific to DBAPIs:
220
+
221
+ * ``isolation_level`` - Set the transaction isolation level for the lifespan
222
+ of a :class:`_engine.Connection` (can only be set on a connection,
223
+ not a statement
224
+ or query). See :ref:`psycopg2_isolation_level`.
225
+
226
+ * ``stream_results`` - Enable or disable usage of psycopg2 server side
227
+ cursors - this feature makes use of "named" cursors in combination with
228
+ special result handling methods so that result rows are not fully buffered.
229
+ Defaults to False, meaning cursors are buffered by default.
230
+
231
+ * ``max_row_buffer`` - when using ``stream_results``, an integer value that
232
+ specifies the maximum number of rows to buffer at a time. This is
233
+ interpreted by the :class:`.BufferedRowCursorResult`, and if omitted the
234
+ buffer will grow to ultimately store 1000 rows at a time.
235
+
236
+ .. versionchanged:: 1.4 The ``max_row_buffer`` size can now be greater than
237
+ 1000, and the buffer will grow to that size.
238
+
239
+ .. _psycopg2_batch_mode:
240
+
241
+ .. _psycopg2_executemany_mode:
242
+
243
+ Psycopg2 Fast Execution Helpers
244
+ -------------------------------
245
+
246
+ Modern versions of psycopg2 include a feature known as
247
+ `Fast Execution Helpers \
248
+ <https://www.psycopg.org/docs/extras.html#fast-execution-helpers>`_, which
249
+ have been shown in benchmarking to improve psycopg2's executemany()
250
+ performance, primarily with INSERT statements, by at least
251
+ an order of magnitude.
252
+
253
+ SQLAlchemy implements a native form of the "insert many values"
254
+ handler that will rewrite a single-row INSERT statement to accommodate for
255
+ many values at once within an extended VALUES clause; this handler is
256
+ equivalent to psycopg2's ``execute_values()`` handler; an overview of this
257
+ feature and its configuration are at :ref:`engine_insertmanyvalues`.
258
+
259
+ .. versionadded:: 2.0 Replaced psycopg2's ``execute_values()`` fast execution
260
+ helper with a native SQLAlchemy mechanism known as
261
+ :ref:`insertmanyvalues <engine_insertmanyvalues>`.
262
+
263
+ The psycopg2 dialect retains the ability to use the psycopg2-specific
264
+ ``execute_batch()`` feature, although it is not expected that this is a widely
265
+ used feature. The use of this extension may be enabled using the
266
+ ``executemany_mode`` flag which may be passed to :func:`_sa.create_engine`::
267
+
268
+ engine = create_engine(
269
+ "postgresql+psycopg2://scott:tiger@host/dbname",
270
+ executemany_mode="values_plus_batch",
271
+ )
272
+
273
+ Possible options for ``executemany_mode`` include:
274
+
275
+ * ``values_only`` - this is the default value. SQLAlchemy's native
276
+ :ref:`insertmanyvalues <engine_insertmanyvalues>` handler is used for qualifying
277
+ INSERT statements, assuming
278
+ :paramref:`_sa.create_engine.use_insertmanyvalues` is left at
279
+ its default value of ``True``. This handler rewrites simple
280
+ INSERT statements to include multiple VALUES clauses so that many
281
+ parameter sets can be inserted with one statement.
282
+
283
+ * ``'values_plus_batch'``- SQLAlchemy's native
284
+ :ref:`insertmanyvalues <engine_insertmanyvalues>` handler is used for qualifying
285
+ INSERT statements, assuming
286
+ :paramref:`_sa.create_engine.use_insertmanyvalues` is left at its default
287
+ value of ``True``. Then, psycopg2's ``execute_batch()`` handler is used for
288
+ qualifying UPDATE and DELETE statements when executed with multiple parameter
289
+ sets. When using this mode, the :attr:`_engine.CursorResult.rowcount`
290
+ attribute will not contain a value for executemany-style executions against
291
+ UPDATE and DELETE statements.
292
+
293
+ .. versionchanged:: 2.0 Removed the ``'batch'`` and ``'None'`` options
294
+ from psycopg2 ``executemany_mode``. Control over batching for INSERT
295
+ statements is now configured via the
296
+ :paramref:`_sa.create_engine.use_insertmanyvalues` engine-level parameter.
297
+
298
+ The term "qualifying statements" refers to the statement being executed
299
+ being a Core :func:`_expression.insert`, :func:`_expression.update`
300
+ or :func:`_expression.delete` construct, and **not** a plain textual SQL
301
+ string or one constructed using :func:`_expression.text`. It also may **not** be
302
+ a special "extension" statement such as an "ON CONFLICT" "upsert" statement.
303
+ When using the ORM, all insert/update/delete statements used by the ORM flush process
304
+ are qualifying.
305
+
306
+ The "page size" for the psycopg2 "batch" strategy can be affected
307
+ by using the ``executemany_batch_page_size`` parameter, which defaults to
308
+ 100.
309
+
310
+ For the "insertmanyvalues" feature, the page size can be controlled using the
311
+ :paramref:`_sa.create_engine.insertmanyvalues_page_size` parameter,
312
+ which defaults to 1000. An example of modifying both parameters
313
+ is below::
314
+
315
+ engine = create_engine(
316
+ "postgresql+psycopg2://scott:tiger@host/dbname",
317
+ executemany_mode="values_plus_batch",
318
+ insertmanyvalues_page_size=5000,
319
+ executemany_batch_page_size=500,
320
+ )
321
+
322
+ .. seealso::
323
+
324
+ :ref:`engine_insertmanyvalues` - background on "insertmanyvalues"
325
+
326
+ :ref:`tutorial_multiple_parameters` - General information on using the
327
+ :class:`_engine.Connection`
328
+ object to execute statements in such a way as to make
329
+ use of the DBAPI ``.executemany()`` method.
330
+
331
+
332
+ .. _psycopg2_unicode:
333
+
334
+ Unicode with Psycopg2
335
+ ----------------------
336
+
337
+ The psycopg2 DBAPI driver supports Unicode data transparently.
338
+
339
+ The client character encoding can be controlled for the psycopg2 dialect
340
+ in the following ways:
341
+
342
+ * For PostgreSQL 9.1 and above, the ``client_encoding`` parameter may be
343
+ passed in the database URL; this parameter is consumed by the underlying
344
+ ``libpq`` PostgreSQL client library::
345
+
346
+ engine = create_engine(
347
+ "postgresql+psycopg2://user:pass@host/dbname?client_encoding=utf8"
348
+ )
349
+
350
+ Alternatively, the above ``client_encoding`` value may be passed using
351
+ :paramref:`_sa.create_engine.connect_args` for programmatic establishment with
352
+ ``libpq``::
353
+
354
+ engine = create_engine(
355
+ "postgresql+psycopg2://user:pass@host/dbname",
356
+ connect_args={"client_encoding": "utf8"},
357
+ )
358
+
359
+ * For all PostgreSQL versions, psycopg2 supports a client-side encoding
360
+ value that will be passed to database connections when they are first
361
+ established. The SQLAlchemy psycopg2 dialect supports this using the
362
+ ``client_encoding`` parameter passed to :func:`_sa.create_engine`::
363
+
364
+ engine = create_engine(
365
+ "postgresql+psycopg2://user:pass@host/dbname", client_encoding="utf8"
366
+ )
367
+
368
+ .. tip:: The above ``client_encoding`` parameter admittedly is very similar
369
+ in appearance to usage of the parameter within the
370
+ :paramref:`_sa.create_engine.connect_args` dictionary; the difference
371
+ above is that the parameter is consumed by psycopg2 and is
372
+ passed to the database connection using ``SET client_encoding TO
373
+ 'utf8'``; in the previously mentioned style, the parameter is instead
374
+ passed through psycopg2 and consumed by the ``libpq`` library.
375
+
376
+ * A common way to set up client encoding with PostgreSQL databases is to
377
+ ensure it is configured within the server-side postgresql.conf file;
378
+ this is the recommended way to set encoding for a server that is
379
+ consistently of one encoding in all databases::
380
+
381
+ # postgresql.conf file
382
+
383
+ # client_encoding = sql_ascii # actually, defaults to database
384
+ # encoding
385
+ client_encoding = utf8
386
+
387
+ Transactions
388
+ ------------
389
+
390
+ The psycopg2 dialect fully supports SAVEPOINT and two-phase commit operations.
391
+
392
+ .. _psycopg2_isolation_level:
393
+
394
+ Psycopg2 Transaction Isolation Level
395
+ -------------------------------------
396
+
397
+ As discussed in :ref:`postgresql_isolation_level`,
398
+ all PostgreSQL dialects support setting of transaction isolation level
399
+ both via the ``isolation_level`` parameter passed to :func:`_sa.create_engine`
400
+ ,
401
+ as well as the ``isolation_level`` argument used by
402
+ :meth:`_engine.Connection.execution_options`. When using the psycopg2 dialect
403
+ , these
404
+ options make use of psycopg2's ``set_isolation_level()`` connection method,
405
+ rather than emitting a PostgreSQL directive; this is because psycopg2's
406
+ API-level setting is always emitted at the start of each transaction in any
407
+ case.
408
+
409
+ The psycopg2 dialect supports these constants for isolation level:
410
+
411
+ * ``READ COMMITTED``
412
+ * ``READ UNCOMMITTED``
413
+ * ``REPEATABLE READ``
414
+ * ``SERIALIZABLE``
415
+ * ``AUTOCOMMIT``
416
+
417
+ .. seealso::
418
+
419
+ :ref:`postgresql_isolation_level`
420
+
421
+ :ref:`pg8000_isolation_level`
422
+
423
+
424
+ NOTICE logging
425
+ ---------------
426
+
427
+ The psycopg2 dialect will log PostgreSQL NOTICE messages
428
+ via the ``sqlalchemy.dialects.postgresql`` logger. When this logger
429
+ is set to the ``logging.INFO`` level, notice messages will be logged::
430
+
431
+ import logging
432
+
433
+ logging.getLogger("sqlalchemy.dialects.postgresql").setLevel(logging.INFO)
434
+
435
+ Above, it is assumed that logging is configured externally. If this is not
436
+ the case, configuration such as ``logging.basicConfig()`` must be utilized::
437
+
438
+ import logging
439
+
440
+ logging.basicConfig() # log messages to stdout
441
+ logging.getLogger("sqlalchemy.dialects.postgresql").setLevel(logging.INFO)
442
+
443
+ .. seealso::
444
+
445
+ `Logging HOWTO <https://docs.python.org/3/howto/logging.html>`_ - on the python.org website
446
+
447
+ .. _psycopg2_hstore:
448
+
449
+ HSTORE type
450
+ ------------
451
+
452
+ The ``psycopg2`` DBAPI includes an extension to natively handle marshalling of
453
+ the HSTORE type. The SQLAlchemy psycopg2 dialect will enable this extension
454
+ by default when psycopg2 version 2.4 or greater is used, and
455
+ it is detected that the target database has the HSTORE type set up for use.
456
+ In other words, when the dialect makes the first
457
+ connection, a sequence like the following is performed:
458
+
459
+ 1. Request the available HSTORE oids using
460
+ ``psycopg2.extras.HstoreAdapter.get_oids()``.
461
+ If this function returns a list of HSTORE identifiers, we then determine
462
+ that the ``HSTORE`` extension is present.
463
+ This function is **skipped** if the version of psycopg2 installed is
464
+ less than version 2.4.
465
+
466
+ 2. If the ``use_native_hstore`` flag is at its default of ``True``, and
467
+ we've detected that ``HSTORE`` oids are available, the
468
+ ``psycopg2.extensions.register_hstore()`` extension is invoked for all
469
+ connections.
470
+
471
+ The ``register_hstore()`` extension has the effect of **all Python
472
+ dictionaries being accepted as parameters regardless of the type of target
473
+ column in SQL**. The dictionaries are converted by this extension into a
474
+ textual HSTORE expression. If this behavior is not desired, disable the
475
+ use of the hstore extension by setting ``use_native_hstore`` to ``False`` as
476
+ follows::
477
+
478
+ engine = create_engine(
479
+ "postgresql+psycopg2://scott:tiger@localhost/test",
480
+ use_native_hstore=False,
481
+ )
482
+
483
+ The ``HSTORE`` type is **still supported** when the
484
+ ``psycopg2.extensions.register_hstore()`` extension is not used. It merely
485
+ means that the coercion between Python dictionaries and the HSTORE
486
+ string format, on both the parameter side and the result side, will take
487
+ place within SQLAlchemy's own marshalling logic, and not that of ``psycopg2``
488
+ which may be more performant.
489
+
490
+ """ # noqa
491
+ from __future__ import annotations
492
+
493
+ import collections.abc as collections_abc
494
+ import logging
495
+ import re
496
+ from typing import cast
497
+
498
+ from . import ranges
499
+ from ._psycopg_common import _PGDialect_common_psycopg
500
+ from ._psycopg_common import _PGExecutionContext_common_psycopg
501
+ from .base import PGIdentifierPreparer
502
+ from .json import JSON
503
+ from .json import JSONB
504
+ from ... import types as sqltypes
505
+ from ... import util
506
+ from ...util import FastIntFlag
507
+ from ...util import parse_user_argument_for_enum
508
+
509
+ logger = logging.getLogger("sqlalchemy.dialects.postgresql")
510
+
511
+
512
+ class _PGJSON(JSON):
513
+ def result_processor(self, dialect, coltype):
514
+ return None
515
+
516
+
517
+ class _PGJSONB(JSONB):
518
+ def result_processor(self, dialect, coltype):
519
+ return None
520
+
521
+
522
+ class _Psycopg2Range(ranges.AbstractSingleRangeImpl):
523
+ _psycopg2_range_cls = "none"
524
+
525
+ def bind_processor(self, dialect):
526
+ psycopg2_Range = getattr(
527
+ cast(PGDialect_psycopg2, dialect)._psycopg2_extras,
528
+ self._psycopg2_range_cls,
529
+ )
530
+
531
+ def to_range(value):
532
+ if isinstance(value, ranges.Range):
533
+ value = psycopg2_Range(
534
+ value.lower, value.upper, value.bounds, value.empty
535
+ )
536
+ return value
537
+
538
+ return to_range
539
+
540
+ def result_processor(self, dialect, coltype):
541
+ def to_range(value):
542
+ if value is not None:
543
+ value = ranges.Range(
544
+ value._lower,
545
+ value._upper,
546
+ bounds=value._bounds if value._bounds else "[)",
547
+ empty=not value._bounds,
548
+ )
549
+ return value
550
+
551
+ return to_range
552
+
553
+
554
+ class _Psycopg2NumericRange(_Psycopg2Range):
555
+ _psycopg2_range_cls = "NumericRange"
556
+
557
+
558
+ class _Psycopg2DateRange(_Psycopg2Range):
559
+ _psycopg2_range_cls = "DateRange"
560
+
561
+
562
+ class _Psycopg2DateTimeRange(_Psycopg2Range):
563
+ _psycopg2_range_cls = "DateTimeRange"
564
+
565
+
566
+ class _Psycopg2DateTimeTZRange(_Psycopg2Range):
567
+ _psycopg2_range_cls = "DateTimeTZRange"
568
+
569
+
570
+ class PGExecutionContext_psycopg2(_PGExecutionContext_common_psycopg):
571
+ _psycopg2_fetched_rows = None
572
+
573
+ def post_exec(self):
574
+ self._log_notices(self.cursor)
575
+
576
+ def _log_notices(self, cursor):
577
+ # check also that notices is an iterable, after it's already
578
+ # established that we will be iterating through it. This is to get
579
+ # around test suites such as SQLAlchemy's using a Mock object for
580
+ # cursor
581
+ if not cursor.connection.notices or not isinstance(
582
+ cursor.connection.notices, collections_abc.Iterable
583
+ ):
584
+ return
585
+
586
+ for notice in cursor.connection.notices:
587
+ # NOTICE messages have a
588
+ # newline character at the end
589
+ logger.info(notice.rstrip())
590
+
591
+ cursor.connection.notices[:] = []
592
+
593
+
594
+ class PGIdentifierPreparer_psycopg2(PGIdentifierPreparer):
595
+ pass
596
+
597
+
598
+ class ExecutemanyMode(FastIntFlag):
599
+ EXECUTEMANY_VALUES = 0
600
+ EXECUTEMANY_VALUES_PLUS_BATCH = 1
601
+
602
+
603
+ (
604
+ EXECUTEMANY_VALUES,
605
+ EXECUTEMANY_VALUES_PLUS_BATCH,
606
+ ) = ExecutemanyMode.__members__.values()
607
+
608
+
609
+ class PGDialect_psycopg2(_PGDialect_common_psycopg):
610
+ driver = "psycopg2"
611
+
612
+ supports_statement_cache = True
613
+ supports_server_side_cursors = True
614
+
615
+ default_paramstyle = "pyformat"
616
+ # set to true based on psycopg2 version
617
+ supports_sane_multi_rowcount = False
618
+ execution_ctx_cls = PGExecutionContext_psycopg2
619
+ preparer = PGIdentifierPreparer_psycopg2
620
+ psycopg2_version = (0, 0)
621
+ use_insertmanyvalues_wo_returning = True
622
+
623
+ returns_native_bytes = False
624
+
625
+ _has_native_hstore = True
626
+
627
+ colspecs = util.update_copy(
628
+ _PGDialect_common_psycopg.colspecs,
629
+ {
630
+ JSON: _PGJSON,
631
+ sqltypes.JSON: _PGJSON,
632
+ JSONB: _PGJSONB,
633
+ ranges.INT4RANGE: _Psycopg2NumericRange,
634
+ ranges.INT8RANGE: _Psycopg2NumericRange,
635
+ ranges.NUMRANGE: _Psycopg2NumericRange,
636
+ ranges.DATERANGE: _Psycopg2DateRange,
637
+ ranges.TSRANGE: _Psycopg2DateTimeRange,
638
+ ranges.TSTZRANGE: _Psycopg2DateTimeTZRange,
639
+ },
640
+ )
641
+
642
+ def __init__(
643
+ self,
644
+ executemany_mode="values_only",
645
+ executemany_batch_page_size=100,
646
+ **kwargs,
647
+ ):
648
+ _PGDialect_common_psycopg.__init__(self, **kwargs)
649
+
650
+ if self._native_inet_types:
651
+ raise NotImplementedError(
652
+ "The psycopg2 dialect does not implement "
653
+ "ipaddress type handling; native_inet_types cannot be set "
654
+ "to ``True`` when using this dialect."
655
+ )
656
+
657
+ # Parse executemany_mode argument, allowing it to be only one of the
658
+ # symbol names
659
+ self.executemany_mode = parse_user_argument_for_enum(
660
+ executemany_mode,
661
+ {
662
+ EXECUTEMANY_VALUES: ["values_only"],
663
+ EXECUTEMANY_VALUES_PLUS_BATCH: ["values_plus_batch"],
664
+ },
665
+ "executemany_mode",
666
+ )
667
+
668
+ self.executemany_batch_page_size = executemany_batch_page_size
669
+
670
+ if self.dbapi and hasattr(self.dbapi, "__version__"):
671
+ m = re.match(r"(\d+)\.(\d+)(?:\.(\d+))?", self.dbapi.__version__)
672
+ if m:
673
+ self.psycopg2_version = tuple(
674
+ int(x) for x in m.group(1, 2, 3) if x is not None
675
+ )
676
+
677
+ if self.psycopg2_version < (2, 7):
678
+ raise ImportError(
679
+ "psycopg2 version 2.7 or higher is required."
680
+ )
681
+
682
+ def initialize(self, connection):
683
+ super().initialize(connection)
684
+ self._has_native_hstore = (
685
+ self.use_native_hstore
686
+ and self._hstore_oids(connection.connection.dbapi_connection)
687
+ is not None
688
+ )
689
+
690
+ self.supports_sane_multi_rowcount = (
691
+ self.executemany_mode is not EXECUTEMANY_VALUES_PLUS_BATCH
692
+ )
693
+
694
+ @classmethod
695
+ def import_dbapi(cls):
696
+ import psycopg2
697
+
698
+ return psycopg2
699
+
700
+ @util.memoized_property
701
+ def _psycopg2_extensions(cls):
702
+ from psycopg2 import extensions
703
+
704
+ return extensions
705
+
706
+ @util.memoized_property
707
+ def _psycopg2_extras(cls):
708
+ from psycopg2 import extras
709
+
710
+ return extras
711
+
712
+ @util.memoized_property
713
+ def _isolation_lookup(self):
714
+ extensions = self._psycopg2_extensions
715
+ return {
716
+ "AUTOCOMMIT": extensions.ISOLATION_LEVEL_AUTOCOMMIT,
717
+ "READ COMMITTED": extensions.ISOLATION_LEVEL_READ_COMMITTED,
718
+ "READ UNCOMMITTED": extensions.ISOLATION_LEVEL_READ_UNCOMMITTED,
719
+ "REPEATABLE READ": extensions.ISOLATION_LEVEL_REPEATABLE_READ,
720
+ "SERIALIZABLE": extensions.ISOLATION_LEVEL_SERIALIZABLE,
721
+ }
722
+
723
+ def set_isolation_level(self, dbapi_connection, level):
724
+ dbapi_connection.set_isolation_level(self._isolation_lookup[level])
725
+
726
+ def set_readonly(self, connection, value):
727
+ connection.readonly = value
728
+
729
+ def get_readonly(self, connection):
730
+ return connection.readonly
731
+
732
+ def set_deferrable(self, connection, value):
733
+ connection.deferrable = value
734
+
735
+ def get_deferrable(self, connection):
736
+ return connection.deferrable
737
+
738
+ def on_connect(self):
739
+ extras = self._psycopg2_extras
740
+
741
+ fns = []
742
+ if self.client_encoding is not None:
743
+
744
+ def on_connect(dbapi_conn):
745
+ dbapi_conn.set_client_encoding(self.client_encoding)
746
+
747
+ fns.append(on_connect)
748
+
749
+ if self.dbapi:
750
+
751
+ def on_connect(dbapi_conn):
752
+ extras.register_uuid(None, dbapi_conn)
753
+
754
+ fns.append(on_connect)
755
+
756
+ if self.dbapi and self.use_native_hstore:
757
+
758
+ def on_connect(dbapi_conn):
759
+ hstore_oids = self._hstore_oids(dbapi_conn)
760
+ if hstore_oids is not None:
761
+ oid, array_oid = hstore_oids
762
+ kw = {"oid": oid}
763
+ kw["array_oid"] = array_oid
764
+ extras.register_hstore(dbapi_conn, **kw)
765
+
766
+ fns.append(on_connect)
767
+
768
+ if self.dbapi and self._json_deserializer:
769
+
770
+ def on_connect(dbapi_conn):
771
+ extras.register_default_json(
772
+ dbapi_conn, loads=self._json_deserializer
773
+ )
774
+ extras.register_default_jsonb(
775
+ dbapi_conn, loads=self._json_deserializer
776
+ )
777
+
778
+ fns.append(on_connect)
779
+
780
+ if fns:
781
+
782
+ def on_connect(dbapi_conn):
783
+ for fn in fns:
784
+ fn(dbapi_conn)
785
+
786
+ return on_connect
787
+ else:
788
+ return None
789
+
790
+ def do_executemany(self, cursor, statement, parameters, context=None):
791
+ if self.executemany_mode is EXECUTEMANY_VALUES_PLUS_BATCH:
792
+ if self.executemany_batch_page_size:
793
+ kwargs = {"page_size": self.executemany_batch_page_size}
794
+ else:
795
+ kwargs = {}
796
+ self._psycopg2_extras.execute_batch(
797
+ cursor, statement, parameters, **kwargs
798
+ )
799
+ else:
800
+ cursor.executemany(statement, parameters)
801
+
802
+ def do_begin_twophase(self, connection, xid):
803
+ connection.connection.tpc_begin(xid)
804
+
805
+ def do_prepare_twophase(self, connection, xid):
806
+ connection.connection.tpc_prepare()
807
+
808
+ def _do_twophase(self, dbapi_conn, operation, xid, recover=False):
809
+ if recover:
810
+ if dbapi_conn.status != self._psycopg2_extensions.STATUS_READY:
811
+ dbapi_conn.rollback()
812
+ operation(xid)
813
+ else:
814
+ operation()
815
+
816
+ def do_rollback_twophase(
817
+ self, connection, xid, is_prepared=True, recover=False
818
+ ):
819
+ dbapi_conn = connection.connection.dbapi_connection
820
+ self._do_twophase(
821
+ dbapi_conn, dbapi_conn.tpc_rollback, xid, recover=recover
822
+ )
823
+
824
+ def do_commit_twophase(
825
+ self, connection, xid, is_prepared=True, recover=False
826
+ ):
827
+ dbapi_conn = connection.connection.dbapi_connection
828
+ self._do_twophase(
829
+ dbapi_conn, dbapi_conn.tpc_commit, xid, recover=recover
830
+ )
831
+
832
+ @util.memoized_instancemethod
833
+ def _hstore_oids(self, dbapi_connection):
834
+ extras = self._psycopg2_extras
835
+ oids = extras.HstoreAdapter.get_oids(dbapi_connection)
836
+ if oids is not None and oids[0]:
837
+ return oids[0:2]
838
+ else:
839
+ return None
840
+
841
+ def is_disconnect(self, e, connection, cursor):
842
+ if isinstance(e, self.dbapi.Error):
843
+ # check the "closed" flag. this might not be
844
+ # present on old psycopg2 versions. Also,
845
+ # this flag doesn't actually help in a lot of disconnect
846
+ # situations, so don't rely on it.
847
+ if getattr(connection, "closed", False):
848
+ return True
849
+
850
+ # checks based on strings. in the case that .closed
851
+ # didn't cut it, fall back onto these.
852
+ str_e = str(e).partition("\n")[0]
853
+ for msg in self._is_disconnect_messages:
854
+ idx = str_e.find(msg)
855
+ if idx >= 0 and '"' not in str_e[:idx]:
856
+ return True
857
+ return False
858
+
859
+ @util.memoized_property
860
+ def _is_disconnect_messages(self):
861
+ return (
862
+ # these error messages from libpq: interfaces/libpq/fe-misc.c
863
+ # and interfaces/libpq/fe-secure.c.
864
+ "terminating connection",
865
+ "closed the connection",
866
+ "connection not open",
867
+ "could not receive data from server",
868
+ "could not send data to server",
869
+ # psycopg2 client errors, psycopg2/connection.h,
870
+ # psycopg2/cursor.h
871
+ "connection already closed",
872
+ "cursor already closed",
873
+ # not sure where this path is originally from, it may
874
+ # be obsolete. It really says "losed", not "closed".
875
+ "losed the connection unexpectedly",
876
+ # these can occur in newer SSL
877
+ "connection has been closed unexpectedly",
878
+ "SSL error: decryption failed or bad record mac",
879
+ "SSL SYSCALL error: Bad file descriptor",
880
+ "SSL SYSCALL error: EOF detected",
881
+ "SSL SYSCALL error: Operation timed out",
882
+ "SSL SYSCALL error: Bad address",
883
+ # This can occur in OpenSSL 1 when an unexpected EOF occurs.
884
+ # https://www.openssl.org/docs/man1.1.1/man3/SSL_get_error.html#BUGS
885
+ # It may also occur in newer OpenSSL for a non-recoverable I/O
886
+ # error as a result of a system call that does not set 'errno'
887
+ # in libc.
888
+ "SSL SYSCALL error: Success",
889
+ )
890
+
891
+
892
+ dialect = PGDialect_psycopg2