import re
-from sqlalchemy.dialects.mysql.base import MySQLDialect, MySQLExecutionContext,\
- MySQLCompiler, MySQLIdentifierPreparer
+from sqlalchemy.dialects.mysql.base import (MySQLDialect,
+ MySQLExecutionContext, MySQLCompiler, MySQLIdentifierPreparer,
+ BIT, NUMERIC)
from sqlalchemy.engine import base as engine_base, default
from sqlalchemy.sql import operators as sql_operators
from sqlalchemy import exc, log, schema, sql, types as sqltypes, util
+from sqlalchemy import processors
class MySQL_mysqlconnectorExecutionContext(MySQLExecutionContext):
value = value.replace(self.escape_quote, self.escape_to_quote)
return value.replace("%", "%%")
+class _myconnpyNumeric(NUMERIC):
+ def result_processor(self, dialect, coltype):
+ if self.asdecimal:
+ return None
+ return processors.to_float
+
+class _myconnpyBIT(BIT):
+ def result_processor(self, dialect, coltype):
+ """MySQL-connector already converts mysql bits, so."""
+
+ return None
+
class MySQL_mysqlconnector(MySQLDialect):
driver = 'mysqlconnector'
supports_unicode_statements = False
- supports_sane_rowcount = True
+ supports_unicode_binds = True
+ supports_sane_rowcount = False
supports_sane_multi_rowcount = True
default_paramstyle = 'format'
preparer = MySQL_mysqlconnectorIdentifierPreparer
+ colspecs = util.update_copy(
+ MySQLDialect.colspecs,
+ {
+ sqltypes.Numeric: _myconnpyNumeric,
+ BIT: _myconnpyBIT,
+ }
+ )
+
@classmethod
def dbapi(cls):
from mysql import connector
def create_connect_args(self, url):
opts = url.translate_connect_args(username='user')
opts.update(url.query)
+
+ util.coerce_kw_type(opts, 'buffered', bool)
+ util.coerce_kw_type(opts, 'raise_on_warnings', bool)
+ opts['buffered'] = True
+ opts['raise_on_warnings'] = True
+
return [[], opts]
def _get_server_version_info(self, connection):
dbapi_con = connection.connection
+
+ from mysql.connector.constants import ClientFlag
+ dbapi_con.set_client_flag(ClientFlag.FOUND_ROWS)
+
version = dbapi_con.get_server_version()
return tuple(version)
def _detect_charset(self, connection):
- """Sniff out the character set in use for connection results."""
-
return connection.connection.get_characterset_info()
def _extract_error_code(self, exception):
- m = re.compile(r"\(.*\)\s+(\d+)").search(str(exception))
- c = m.group(1)
- if c:
- return int(c)
- else:
+ try:
+ return exception.orig.errno
+ except AttributeError:
return None
+
+ def is_disconnect(self, e):
+ errnos = (2006, 2013, 2014, 2045, 2055, 2048)
+ exceptions = (self.dbapi.OperationalError,self.dbapi.InterfaceError)
+ if isinstance(e, exceptions):
+ return e.errno in errnos
+ else:
+ return False
+
+ def _compat_fetchall(self, rp, charset=None):
+ return rp.fetchall()
+ def _compat_fetchone(self, rp, charset=None):
+ return rp.fetchone()
+
dialect = MySQL_mysqlconnector
def teardown_class(cls):
metadata.drop_all()
+ @testing.fails_on('mysql+mysqlconnector', 'uses pyformat')
def test_expression(self):
format = testing.db.dialect.paramstyle == 'format' and '%s' or '?'
self.assert_compile(
matchtable.c.title.match('somstr'),
"MATCH (matchtable.title) AGAINST (%s IN BOOLEAN MODE)" % format)
+
+ @testing.fails_on('mysql+mysqldb', 'uses format')
+ @testing.fails_on('mysql+oursql', 'uses format')
+ @testing.fails_on('mysql+pyodbc', 'uses format')
+ @testing.fails_on('mysql+zxjdbc', 'uses format')
+ def test_expression(self):
+ format = '%(title_1)s'
+ self.assert_compile(
+ matchtable.c.title.match('somstr'),
+ "MATCH (matchtable.title) AGAINST (%s IN BOOLEAN MODE)" % format)
def test_simple_match(self):
results = (matchtable.select().