diff options
author | Mike Bayer <mike_mp@zzzcomputing.com> | 2015-05-15 12:35:21 -0400 |
---|---|---|
committer | Mike Bayer <mike_mp@zzzcomputing.com> | 2015-05-15 12:38:28 -0400 |
commit | d178707ecaeb547470e2b7b37b9a939abc69cbd0 (patch) | |
tree | dc4b674748cf673a123b90797c3525c3a13ef657 | |
parent | 64c1f2e5688879e1cc087f4e4a893d32267fd1fb (diff) | |
download | sqlalchemy-d178707ecaeb547470e2b7b37b9a939abc69cbd0.tar.gz |
- Added support for the case of the misbehaving DBAPI that has
pep-249 exception names linked to exception classes of an entirely
different name, preventing SQLAlchemy's own exception wrapping from
wrapping the error appropriately.
The SQLAlchemy dialect in use needs to implement a new
accessor :attr:`.DefaultDialect.dbapi_exception_translation_map`
to support this feature; this is implemented now for the py-postgresql
dialect.
fixes #3421
-rw-r--r-- | doc/build/changelog/changelog_10.rst | 13 | ||||
-rw-r--r-- | lib/sqlalchemy/dialects/postgresql/pypostgresql.py | 17 | ||||
-rw-r--r-- | lib/sqlalchemy/engine/base.py | 6 | ||||
-rw-r--r-- | lib/sqlalchemy/engine/default.py | 9 | ||||
-rw-r--r-- | lib/sqlalchemy/engine/interfaces.py | 10 | ||||
-rw-r--r-- | lib/sqlalchemy/exc.py | 8 | ||||
-rw-r--r-- | lib/sqlalchemy/testing/suite/__init__.py | 1 | ||||
-rw-r--r-- | lib/sqlalchemy/testing/suite/test_dialect.py | 39 | ||||
-rw-r--r-- | test/base/test_except.py | 57 |
9 files changed, 155 insertions, 5 deletions
diff --git a/doc/build/changelog/changelog_10.rst b/doc/build/changelog/changelog_10.rst index 0a012962f..f4091a988 100644 --- a/doc/build/changelog/changelog_10.rst +++ b/doc/build/changelog/changelog_10.rst @@ -19,6 +19,19 @@ :version: 1.0.5 .. change:: + :tags: bug, engine + :tickets: 3421 + + Added support for the case of the misbehaving DBAPI that has + pep-249 exception names linked to exception classes of an entirely + different name, preventing SQLAlchemy's own exception wrapping from + wrapping the error appropriately. + The SQLAlchemy dialect in use needs to implement a new + accessor :attr:`.DefaultDialect.dbapi_exception_translation_map` + to support this feature; this is implemented now for the py-postgresql + dialect. + + .. change:: :tags: bug, orm :tickets: 3420 diff --git a/lib/sqlalchemy/dialects/postgresql/pypostgresql.py b/lib/sqlalchemy/dialects/postgresql/pypostgresql.py index 00c67d170..db6d5e16c 100644 --- a/lib/sqlalchemy/dialects/postgresql/pypostgresql.py +++ b/lib/sqlalchemy/dialects/postgresql/pypostgresql.py @@ -65,6 +65,23 @@ class PGDialect_pypostgresql(PGDialect): from postgresql.driver import dbapi20 return dbapi20 + _DBAPI_ERROR_NAMES = [ + "Error", + "InterfaceError", "DatabaseError", "DataError", + "OperationalError", "IntegrityError", "InternalError", + "ProgrammingError", "NotSupportedError" + ] + + @util.memoized_property + def dbapi_exception_translation_map(self): + if self.dbapi is None: + return {} + + return dict( + (getattr(self.dbapi, name).__name__, name) + for name in self._DBAPI_ERROR_NAMES + ) + def create_connect_args(self, url): opts = url.translate_connect_args(username='user') if 'port' in opts: diff --git a/lib/sqlalchemy/engine/base.py b/lib/sqlalchemy/engine/base.py index af310c450..7ebe39bbf 100644 --- a/lib/sqlalchemy/engine/base.py +++ b/lib/sqlalchemy/engine/base.py @@ -1261,7 +1261,8 @@ class Connection(Connectable): exc.DBAPIError.instance(statement, parameters, e, - self.dialect.dbapi.Error), + self.dialect.dbapi.Error, + dialect=self.dialect), exc_info ) self._reentrant_error = True @@ -1277,7 +1278,8 @@ class Connection(Connectable): parameters, e, self.dialect.dbapi.Error, - connection_invalidated=self._is_disconnect) + connection_invalidated=self._is_disconnect, + dialect=self.dialect) else: sqlalchemy_exception = None diff --git a/lib/sqlalchemy/engine/default.py b/lib/sqlalchemy/engine/default.py index 763e85f82..9330a602c 100644 --- a/lib/sqlalchemy/engine/default.py +++ b/lib/sqlalchemy/engine/default.py @@ -157,6 +157,15 @@ class DefaultDialect(interfaces.Dialect): reflection_options = () + dbapi_exception_translation_map = util.immutabledict() + """mapping used in the extremely unusual case that a DBAPI's + published exceptions don't actually have the __name__ that they + are linked towards. + + .. versionadded:: 1.0.5 + + """ + def __init__(self, convert_unicode=False, encoding='utf-8', paramstyle=None, dbapi=None, implicit_returning=None, diff --git a/lib/sqlalchemy/engine/interfaces.py b/lib/sqlalchemy/engine/interfaces.py index 2dd192162..73a8b4635 100644 --- a/lib/sqlalchemy/engine/interfaces.py +++ b/lib/sqlalchemy/engine/interfaces.py @@ -150,6 +150,16 @@ class Dialect(object): This will prevent types.Boolean from generating a CHECK constraint when that type is used. + dbapi_exception_translation_map + A dictionary of names that will contain as values the names of + pep-249 exceptions ("IntegrityError", "OperationalError", etc) + keyed to alternate class names, to support the case where a + DBAPI has exception classes that aren't named as they are + referred to (e.g. IntegrityError = MyException). In the vast + majority of cases this dictionary is empty. + + .. versionadded:: 1.0.5 + """ _has_events = False diff --git a/lib/sqlalchemy/exc.py b/lib/sqlalchemy/exc.py index 9b27436b3..3a4f346e0 100644 --- a/lib/sqlalchemy/exc.py +++ b/lib/sqlalchemy/exc.py @@ -13,8 +13,6 @@ raised as a result of DBAPI exceptions are all subclasses of """ -import traceback - class SQLAlchemyError(Exception): """Generic error class.""" @@ -278,7 +276,8 @@ class DBAPIError(StatementError): @classmethod def instance(cls, statement, params, orig, dbapi_base_err, - connection_invalidated=False): + connection_invalidated=False, + dialect=None): # Don't ever wrap these, just return them directly as if # DBAPIError didn't exist. if (isinstance(orig, BaseException) and @@ -300,6 +299,9 @@ class DBAPIError(StatementError): glob = globals() for super_ in orig.__class__.__mro__: name = super_.__name__ + if dialect: + name = dialect.dbapi_exception_translation_map.get( + name, name) if name in glob and issubclass(glob[name], DBAPIError): cls = glob[name] break diff --git a/lib/sqlalchemy/testing/suite/__init__.py b/lib/sqlalchemy/testing/suite/__init__.py index 780aa40aa..9eeffd4cb 100644 --- a/lib/sqlalchemy/testing/suite/__init__.py +++ b/lib/sqlalchemy/testing/suite/__init__.py @@ -1,4 +1,5 @@ +from sqlalchemy.testing.suite.test_dialect import * from sqlalchemy.testing.suite.test_ddl import * from sqlalchemy.testing.suite.test_insert import * from sqlalchemy.testing.suite.test_sequence import * diff --git a/lib/sqlalchemy/testing/suite/test_dialect.py b/lib/sqlalchemy/testing/suite/test_dialect.py new file mode 100644 index 000000000..5ad5694b3 --- /dev/null +++ b/lib/sqlalchemy/testing/suite/test_dialect.py @@ -0,0 +1,39 @@ +from .. import fixtures, config +from sqlalchemy import exc +from sqlalchemy import Integer, String +from .. import assert_raises +from ..schema import Table, Column + + +class ExceptionTest(fixtures.TablesTest): + """Test basic exception wrapping. + + DBAPIs vary a lot in exception behavior so to actually anticipate + specific exceptions from real round trips, we need to be conservative. + + """ + run_deletes = 'each' + + __backend__ = True + + @classmethod + def define_tables(cls, metadata): + Table('manual_pk', metadata, + Column('id', Integer, primary_key=True, autoincrement=False), + Column('data', String(50)) + ) + + def test_integrity_error(self): + + with config.db.begin() as conn: + conn.execute( + self.tables.manual_pk.insert(), + {'id': 1, 'data': 'd1'} + ) + + assert_raises( + exc.IntegrityError, + conn.execute, + self.tables.manual_pk.insert(), + {'id': 1, 'data': 'd1'} + ) diff --git a/test/base/test_except.py b/test/base/test_except.py index 918e7a042..9e8dd4760 100644 --- a/test/base/test_except.py +++ b/test/base/test_except.py @@ -4,6 +4,7 @@ from sqlalchemy import exc as sa_exceptions from sqlalchemy.testing import fixtures from sqlalchemy.testing import eq_ +from sqlalchemy.engine import default class Error(Exception): @@ -28,8 +29,28 @@ class OutOfSpec(DatabaseError): pass +# exception with a totally different name... +class WrongNameError(DatabaseError): + pass + +# but they're going to call it their "IntegrityError" +IntegrityError = WrongNameError + + +# and they're going to subclass it! +class SpecificIntegrityError(WrongNameError): + pass + + class WrapTest(fixtures.TestBase): + def _translating_dialect_fixture(self): + d = default.DefaultDialect() + d.dbapi_exception_translation_map = { + "WrongNameError": "IntegrityError" + } + return d + def test_db_error_normal(self): try: raise sa_exceptions.DBAPIError.instance( @@ -160,6 +181,42 @@ class WrapTest(fixtures.TestBase): except sa_exceptions.ArgumentError: self.assert_(False) + dialect = self._translating_dialect_fixture() + try: + raise sa_exceptions.DBAPIError.instance( + '', [], + sa_exceptions.ArgumentError(), DatabaseError, + dialect=dialect) + except sa_exceptions.DBAPIError as e: + self.assert_(e.__class__ is sa_exceptions.DBAPIError) + except sa_exceptions.ArgumentError: + self.assert_(False) + + def test_db_error_dbapi_uses_wrong_names(self): + dialect = self._translating_dialect_fixture() + + try: + raise sa_exceptions.DBAPIError.instance( + '', [], IntegrityError(), + DatabaseError, dialect=dialect) + except sa_exceptions.DBAPIError as e: + self.assert_(e.__class__ is sa_exceptions.IntegrityError) + + try: + raise sa_exceptions.DBAPIError.instance( + '', [], SpecificIntegrityError(), + DatabaseError, dialect=dialect) + except sa_exceptions.DBAPIError as e: + self.assert_(e.__class__ is sa_exceptions.IntegrityError) + + try: + raise sa_exceptions.DBAPIError.instance( + '', [], SpecificIntegrityError(), + DatabaseError) + except sa_exceptions.DBAPIError as e: + # doesn't work without a dialect + self.assert_(e.__class__ is not sa_exceptions.IntegrityError) + def test_db_error_keyboard_interrupt(self): try: raise sa_exceptions.DBAPIError.instance( |