You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							241 lines
						
					
					
						
							7.4 KiB
						
					
					
				
			
		
		
	
	
							241 lines
						
					
					
						
							7.4 KiB
						
					
					
				# mysql/mariadbconnector.py
 | 
						|
# Copyright (C) 2005-2022 the SQLAlchemy authors and contributors
 | 
						|
# <see AUTHORS file>
 | 
						|
#
 | 
						|
# This module is part of SQLAlchemy and is released under
 | 
						|
# the MIT License: https://www.opensource.org/licenses/mit-license.php
 | 
						|
 | 
						|
"""
 | 
						|
 | 
						|
.. dialect:: mysql+mariadbconnector
 | 
						|
    :name: MariaDB Connector/Python
 | 
						|
    :dbapi: mariadb
 | 
						|
    :connectstring: mariadb+mariadbconnector://<user>:<password>@<host>[:<port>]/<dbname>
 | 
						|
    :url: https://pypi.org/project/mariadb/
 | 
						|
 | 
						|
Driver Status
 | 
						|
-------------
 | 
						|
 | 
						|
MariaDB Connector/Python enables Python programs to access MariaDB and MySQL
 | 
						|
databases using an API which is compliant with the Python DB API 2.0 (PEP-249).
 | 
						|
It is written in C and uses MariaDB Connector/C client library for client server
 | 
						|
communication.
 | 
						|
 | 
						|
Note that the default driver for a ``mariadb://`` connection URI continues to
 | 
						|
be ``mysqldb``. ``mariadb+mariadbconnector://`` is required to use this driver.
 | 
						|
 | 
						|
.. mariadb: https://github.com/mariadb-corporation/mariadb-connector-python
 | 
						|
 | 
						|
"""  # noqa
 | 
						|
import re
 | 
						|
 | 
						|
from .base import MySQLCompiler
 | 
						|
from .base import MySQLDialect
 | 
						|
from .base import MySQLExecutionContext
 | 
						|
from ... import sql
 | 
						|
from ... import util
 | 
						|
 | 
						|
mariadb_cpy_minimum_version = (1, 0, 1)
 | 
						|
 | 
						|
 | 
						|
class MySQLExecutionContext_mariadbconnector(MySQLExecutionContext):
 | 
						|
    _lastrowid = None
 | 
						|
 | 
						|
    def create_server_side_cursor(self):
 | 
						|
        return self._dbapi_connection.cursor(buffered=False)
 | 
						|
 | 
						|
    def create_default_cursor(self):
 | 
						|
        return self._dbapi_connection.cursor(buffered=True)
 | 
						|
 | 
						|
    def post_exec(self):
 | 
						|
        if self.isinsert and self.compiled.postfetch_lastrowid:
 | 
						|
            self._lastrowid = self.cursor.lastrowid
 | 
						|
 | 
						|
    def get_lastrowid(self):
 | 
						|
        return self._lastrowid
 | 
						|
 | 
						|
 | 
						|
class MySQLCompiler_mariadbconnector(MySQLCompiler):
 | 
						|
    pass
 | 
						|
 | 
						|
 | 
						|
class MySQLDialect_mariadbconnector(MySQLDialect):
 | 
						|
    driver = "mariadbconnector"
 | 
						|
    supports_statement_cache = True
 | 
						|
 | 
						|
    # set this to True at the module level to prevent the driver from running
 | 
						|
    # against a backend that server detects as MySQL. currently this appears to
 | 
						|
    # be unnecessary as MariaDB client libraries have always worked against
 | 
						|
    # MySQL databases.   However, if this changes at some point, this can be
 | 
						|
    # adjusted, but PLEASE ADD A TEST in test/dialect/mysql/test_dialect.py if
 | 
						|
    # this change is made at some point to ensure the correct exception
 | 
						|
    # is raised at the correct point when running the driver against
 | 
						|
    # a MySQL backend.
 | 
						|
    # is_mariadb = True
 | 
						|
 | 
						|
    supports_unicode_statements = True
 | 
						|
    encoding = "utf8mb4"
 | 
						|
    convert_unicode = True
 | 
						|
    supports_sane_rowcount = True
 | 
						|
    supports_sane_multi_rowcount = True
 | 
						|
    supports_native_decimal = True
 | 
						|
    default_paramstyle = "qmark"
 | 
						|
    execution_ctx_cls = MySQLExecutionContext_mariadbconnector
 | 
						|
    statement_compiler = MySQLCompiler_mariadbconnector
 | 
						|
 | 
						|
    supports_server_side_cursors = True
 | 
						|
 | 
						|
    @util.memoized_property
 | 
						|
    def _dbapi_version(self):
 | 
						|
        if self.dbapi and hasattr(self.dbapi, "__version__"):
 | 
						|
            return tuple(
 | 
						|
                [
 | 
						|
                    int(x)
 | 
						|
                    for x in re.findall(
 | 
						|
                        r"(\d+)(?:[-\.]?|$)", self.dbapi.__version__
 | 
						|
                    )
 | 
						|
                ]
 | 
						|
            )
 | 
						|
        else:
 | 
						|
            return (99, 99, 99)
 | 
						|
 | 
						|
    def __init__(self, **kwargs):
 | 
						|
        super(MySQLDialect_mariadbconnector, self).__init__(**kwargs)
 | 
						|
        self.paramstyle = "qmark"
 | 
						|
        if self.dbapi is not None:
 | 
						|
            if self._dbapi_version < mariadb_cpy_minimum_version:
 | 
						|
                raise NotImplementedError(
 | 
						|
                    "The minimum required version for MariaDB "
 | 
						|
                    "Connector/Python is %s"
 | 
						|
                    % ".".join(str(x) for x in mariadb_cpy_minimum_version)
 | 
						|
                )
 | 
						|
 | 
						|
    @classmethod
 | 
						|
    def dbapi(cls):
 | 
						|
        return __import__("mariadb")
 | 
						|
 | 
						|
    def is_disconnect(self, e, connection, cursor):
 | 
						|
        if super(MySQLDialect_mariadbconnector, self).is_disconnect(
 | 
						|
            e, connection, cursor
 | 
						|
        ):
 | 
						|
            return True
 | 
						|
        elif isinstance(e, self.dbapi.Error):
 | 
						|
            str_e = str(e).lower()
 | 
						|
            return "not connected" in str_e or "isn't valid" in str_e
 | 
						|
        else:
 | 
						|
            return False
 | 
						|
 | 
						|
    def create_connect_args(self, url):
 | 
						|
        opts = url.translate_connect_args()
 | 
						|
 | 
						|
        int_params = [
 | 
						|
            "connect_timeout",
 | 
						|
            "read_timeout",
 | 
						|
            "write_timeout",
 | 
						|
            "client_flag",
 | 
						|
            "port",
 | 
						|
            "pool_size",
 | 
						|
        ]
 | 
						|
        bool_params = [
 | 
						|
            "local_infile",
 | 
						|
            "ssl_verify_cert",
 | 
						|
            "ssl",
 | 
						|
            "pool_reset_connection",
 | 
						|
        ]
 | 
						|
 | 
						|
        for key in int_params:
 | 
						|
            util.coerce_kw_type(opts, key, int)
 | 
						|
        for key in bool_params:
 | 
						|
            util.coerce_kw_type(opts, key, bool)
 | 
						|
 | 
						|
        # FOUND_ROWS must be set in CLIENT_FLAGS to enable
 | 
						|
        # supports_sane_rowcount.
 | 
						|
        client_flag = opts.get("client_flag", 0)
 | 
						|
        if self.dbapi is not None:
 | 
						|
            try:
 | 
						|
                CLIENT_FLAGS = __import__(
 | 
						|
                    self.dbapi.__name__ + ".constants.CLIENT"
 | 
						|
                ).constants.CLIENT
 | 
						|
                client_flag |= CLIENT_FLAGS.FOUND_ROWS
 | 
						|
            except (AttributeError, ImportError):
 | 
						|
                self.supports_sane_rowcount = False
 | 
						|
            opts["client_flag"] = client_flag
 | 
						|
        return [[], opts]
 | 
						|
 | 
						|
    def _extract_error_code(self, exception):
 | 
						|
        try:
 | 
						|
            rc = exception.errno
 | 
						|
        except:
 | 
						|
            rc = -1
 | 
						|
        return rc
 | 
						|
 | 
						|
    def _detect_charset(self, connection):
 | 
						|
        return "utf8mb4"
 | 
						|
 | 
						|
    _isolation_lookup = set(
 | 
						|
        [
 | 
						|
            "SERIALIZABLE",
 | 
						|
            "READ UNCOMMITTED",
 | 
						|
            "READ COMMITTED",
 | 
						|
            "REPEATABLE READ",
 | 
						|
            "AUTOCOMMIT",
 | 
						|
        ]
 | 
						|
    )
 | 
						|
 | 
						|
    def _set_isolation_level(self, connection, level):
 | 
						|
        if level == "AUTOCOMMIT":
 | 
						|
            connection.autocommit = True
 | 
						|
        else:
 | 
						|
            connection.autocommit = False
 | 
						|
            super(MySQLDialect_mariadbconnector, self)._set_isolation_level(
 | 
						|
                connection, level
 | 
						|
            )
 | 
						|
 | 
						|
    def do_begin_twophase(self, connection, xid):
 | 
						|
        connection.execute(
 | 
						|
            sql.text("XA BEGIN :xid").bindparams(
 | 
						|
                sql.bindparam("xid", xid, literal_execute=True)
 | 
						|
            )
 | 
						|
        )
 | 
						|
 | 
						|
    def do_prepare_twophase(self, connection, xid):
 | 
						|
        connection.execute(
 | 
						|
            sql.text("XA END :xid").bindparams(
 | 
						|
                sql.bindparam("xid", xid, literal_execute=True)
 | 
						|
            )
 | 
						|
        )
 | 
						|
        connection.execute(
 | 
						|
            sql.text("XA PREPARE :xid").bindparams(
 | 
						|
                sql.bindparam("xid", xid, literal_execute=True)
 | 
						|
            )
 | 
						|
        )
 | 
						|
 | 
						|
    def do_rollback_twophase(
 | 
						|
        self, connection, xid, is_prepared=True, recover=False
 | 
						|
    ):
 | 
						|
        if not is_prepared:
 | 
						|
            connection.execute(
 | 
						|
                sql.text("XA END :xid").bindparams(
 | 
						|
                    sql.bindparam("xid", xid, literal_execute=True)
 | 
						|
                )
 | 
						|
            )
 | 
						|
        connection.execute(
 | 
						|
            sql.text("XA ROLLBACK :xid").bindparams(
 | 
						|
                sql.bindparam("xid", xid, literal_execute=True)
 | 
						|
            )
 | 
						|
        )
 | 
						|
 | 
						|
    def do_commit_twophase(
 | 
						|
        self, connection, xid, is_prepared=True, recover=False
 | 
						|
    ):
 | 
						|
        if not is_prepared:
 | 
						|
            self.do_prepare_twophase(connection, xid)
 | 
						|
        connection.execute(
 | 
						|
            sql.text("XA COMMIT :xid").bindparams(
 | 
						|
                sql.bindparam("xid", xid, literal_execute=True)
 | 
						|
            )
 | 
						|
        )
 | 
						|
 | 
						|
 | 
						|
dialect = MySQLDialect_mariadbconnector
 |