Raise minimum SQLAlchemy version to 1.2.0
Per [1], this is the latest supported version of SQLAlchemy. 1.1.x and earlier are EOL. [1] https://www.sqlalchemy.org/download.html#relstatus Change-Id: I63e4baf772be9ddfb787ac3aff01fcaddf7b901c Signed-off-by: Stephen Finucane <stephenfin@redhat.com>
This commit is contained in:
parent
fafcbddec7
commit
e4f2020b94
@ -55,7 +55,7 @@ requestsexceptions==1.2.0
|
|||||||
rfc3986==0.3.1
|
rfc3986==0.3.1
|
||||||
six==1.10.0
|
six==1.10.0
|
||||||
smmap==0.9.0
|
smmap==0.9.0
|
||||||
SQLAlchemy==1.0.10
|
SQLAlchemy==1.2.0
|
||||||
sqlalchemy-migrate==0.11.0
|
sqlalchemy-migrate==0.11.0
|
||||||
sqlparse==0.2.2
|
sqlparse==0.2.2
|
||||||
stevedore==1.20.0
|
stevedore==1.20.0
|
||||||
|
@ -1,69 +0,0 @@
|
|||||||
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
||||||
# not use this file except in compliance with the License. You may obtain
|
|
||||||
# a copy of the License at
|
|
||||||
#
|
|
||||||
# http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
#
|
|
||||||
# Unless required by applicable law or agreed to in writing, software
|
|
||||||
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
||||||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
||||||
# License for the specific language governing permissions and limitations
|
|
||||||
# under the License.
|
|
||||||
import re
|
|
||||||
|
|
||||||
import sqlalchemy
|
|
||||||
|
|
||||||
|
|
||||||
SQLA_VERSION = tuple(
|
|
||||||
int(num) if re.match(r'^\d+$', num) else num
|
|
||||||
for num in sqlalchemy.__version__.split(".")
|
|
||||||
)
|
|
||||||
|
|
||||||
sqla_110 = SQLA_VERSION >= (1, 1, 0)
|
|
||||||
sqla_100 = SQLA_VERSION >= (1, 0, 0)
|
|
||||||
sqla_097 = SQLA_VERSION >= (0, 9, 7)
|
|
||||||
sqla_094 = SQLA_VERSION >= (0, 9, 4)
|
|
||||||
sqla_090 = SQLA_VERSION >= (0, 9, 0)
|
|
||||||
sqla_08 = SQLA_VERSION >= (0, 8)
|
|
||||||
|
|
||||||
|
|
||||||
def get_postgresql_enums(conn):
|
|
||||||
"""Return a list of ENUM type names on a Postgresql backend.
|
|
||||||
|
|
||||||
For SQLAlchemy 0.9 and lower, makes use of the semi-private
|
|
||||||
_load_enums() method of the Postgresql dialect. In SQLAlchemy
|
|
||||||
1.0 this feature is supported using get_enums().
|
|
||||||
|
|
||||||
This function may only be called when the given connection
|
|
||||||
is against the Postgresql backend. It will fail for other
|
|
||||||
kinds of backends.
|
|
||||||
|
|
||||||
"""
|
|
||||||
if sqla_100:
|
|
||||||
return [e['name'] for e in sqlalchemy.inspect(conn).get_enums()]
|
|
||||||
else:
|
|
||||||
return conn.dialect._load_enums(conn).keys()
|
|
||||||
|
|
||||||
|
|
||||||
def adapt_type_object(type_object, target_class, *args, **kw):
|
|
||||||
"""Call the adapt() method on a type.
|
|
||||||
|
|
||||||
For SQLAlchemy 1.0, runs a local version of constructor_copy() that
|
|
||||||
allows keyword arguments to be overridden.
|
|
||||||
|
|
||||||
See https://github.com/zzzeek/sqlalchemy/commit/\
|
|
||||||
ceeb033054f09db3eccbde3fad1941ec42919a54
|
|
||||||
|
|
||||||
"""
|
|
||||||
if sqla_110:
|
|
||||||
return type_object.adapt(target_class, *args, **kw)
|
|
||||||
else:
|
|
||||||
# NOTE(zzzeek): this only works for basic types, won't work for
|
|
||||||
# schema types like Enum, Boolean
|
|
||||||
# NOTE(zzzeek): this code can be removed once requirements
|
|
||||||
# are at SQLAlchemy >= 1.1
|
|
||||||
names = sqlalchemy.util.get_cls_kwargs(target_class)
|
|
||||||
kw.update(
|
|
||||||
(k, type_object.__dict__[k]) for k in names.difference(kw)
|
|
||||||
if k in type_object.__dict__)
|
|
||||||
return target_class(*args, **kw)
|
|
@ -519,10 +519,5 @@ def register_engine(engine):
|
|||||||
|
|
||||||
|
|
||||||
def handle_connect_error(engine):
|
def handle_connect_error(engine):
|
||||||
"""Connect to the engine, including handle_error handlers.
|
"""Connect to the engine, including handle_error handlers."""
|
||||||
|
|
||||||
The compat library now builds this into the engine.connect()
|
|
||||||
system as per SQLAlchemy 1.0's behavior.
|
|
||||||
|
|
||||||
"""
|
|
||||||
return engine.connect()
|
return engine.connect()
|
||||||
|
@ -15,7 +15,6 @@
|
|||||||
|
|
||||||
import re
|
import re
|
||||||
|
|
||||||
from oslo_db.sqlalchemy.compat import utils as compat_utils
|
|
||||||
from oslo_db.sqlalchemy.types import String
|
from oslo_db.sqlalchemy.types import String
|
||||||
|
|
||||||
from sqlalchemy import event, schema
|
from sqlalchemy import event, schema
|
||||||
@ -100,8 +99,8 @@ def _compile_ndb_string(element, compiler, **kw):
|
|||||||
return compiler.visit_string(element, **kw)
|
return compiler.visit_string(element, **kw)
|
||||||
|
|
||||||
if element.mysql_ndb_length:
|
if element.mysql_ndb_length:
|
||||||
effective_type = compat_utils.adapt_type_object(
|
effective_type = element.adapt(
|
||||||
element, _String, length=element.mysql_ndb_length)
|
_String, length=element.mysql_ndb_length)
|
||||||
return compiler.visit_string(effective_type, **kw)
|
return compiler.visit_string(effective_type, **kw)
|
||||||
elif element.mysql_ndb_type:
|
elif element.mysql_ndb_type:
|
||||||
effective_type = to_instance(element.mysql_ndb_type)
|
effective_type = to_instance(element.mysql_ndb_type)
|
||||||
|
@ -28,7 +28,6 @@ from sqlalchemy.ext.declarative import declarative_base
|
|||||||
from sqlalchemy.orm import mapper
|
from sqlalchemy.orm import mapper
|
||||||
|
|
||||||
from oslo_db import exception
|
from oslo_db import exception
|
||||||
from oslo_db.sqlalchemy.compat import utils as compat_utils
|
|
||||||
from oslo_db.sqlalchemy import engines
|
from oslo_db.sqlalchemy import engines
|
||||||
from oslo_db.sqlalchemy import exc_filters
|
from oslo_db.sqlalchemy import exc_filters
|
||||||
from oslo_db.tests.sqlalchemy import base as test_base
|
from oslo_db.tests.sqlalchemy import base as test_base
|
||||||
@ -655,48 +654,25 @@ class TestExceptionCauseMySQLSavepoint(test_base._MySQLOpportunisticTestCase):
|
|||||||
with session.begin():
|
with session.begin():
|
||||||
session.add(self.A(id=1))
|
session.add(self.A(id=1))
|
||||||
try:
|
try:
|
||||||
|
|
||||||
with session.begin():
|
with session.begin():
|
||||||
|
|
||||||
try:
|
try:
|
||||||
with session.begin_nested():
|
with session.begin_nested():
|
||||||
session.execute("rollback")
|
session.execute("rollback")
|
||||||
session.add(self.A(id=1))
|
session.add(self.A(id=1))
|
||||||
|
|
||||||
# outermost is the failed SAVEPOINT rollback
|
# outermost is the failed SAVEPOINT rollback
|
||||||
# from the "with session.begin_nested()"
|
# from the "with session.begin_nested()"
|
||||||
except exception.DBError as dbe_inner:
|
except exception.DBError as dbe_inner:
|
||||||
|
# in SQLA 1.1+, the rollback() method of Session
|
||||||
if not compat_utils.sqla_110:
|
# catches the error and repairs the state of the
|
||||||
# first "cause" is the failed SAVEPOINT rollback
|
# session even though the SAVEPOINT was lost;
|
||||||
# from inside of flush(), when it fails
|
# the net result here is that one exception is thrown
|
||||||
self.assertTrue(
|
# instead of two. This is SQLAlchemy ticket #3680
|
||||||
isinstance(
|
self.assertTrue(
|
||||||
dbe_inner.cause,
|
isinstance(
|
||||||
exception.DBError
|
dbe_inner.cause,
|
||||||
)
|
exception.DBDuplicateEntry
|
||||||
)
|
)
|
||||||
|
)
|
||||||
# second "cause" is then the actual DB duplicate
|
|
||||||
self.assertTrue(
|
|
||||||
isinstance(
|
|
||||||
dbe_inner.cause.cause,
|
|
||||||
exception.DBDuplicateEntry
|
|
||||||
)
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
# in SQLA 1.1, the rollback() method of Session
|
|
||||||
# catches the error and repairs the state of the
|
|
||||||
# session even though the SAVEPOINT was lost;
|
|
||||||
# the net result here is that one exception is thrown
|
|
||||||
# instead of two. This is SQLAlchemy ticket #3680
|
|
||||||
self.assertTrue(
|
|
||||||
isinstance(
|
|
||||||
dbe_inner.cause,
|
|
||||||
exception.DBDuplicateEntry
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
except exception.DBError as dbe_outer:
|
except exception.DBError as dbe_outer:
|
||||||
self.assertTrue(
|
self.assertTrue(
|
||||||
isinstance(
|
isinstance(
|
||||||
|
@ -23,6 +23,7 @@ from sqlalchemy import Boolean, Index, Integer, DateTime, String, SmallInteger
|
|||||||
from sqlalchemy import CheckConstraint
|
from sqlalchemy import CheckConstraint
|
||||||
from sqlalchemy import MetaData, Table, Column
|
from sqlalchemy import MetaData, Table, Column
|
||||||
from sqlalchemy import ForeignKey, ForeignKeyConstraint
|
from sqlalchemy import ForeignKey, ForeignKeyConstraint
|
||||||
|
from sqlalchemy.dialects.postgresql import psycopg2
|
||||||
from sqlalchemy.engine import reflection
|
from sqlalchemy.engine import reflection
|
||||||
from sqlalchemy.engine import url as sa_url
|
from sqlalchemy.engine import url as sa_url
|
||||||
from sqlalchemy.exc import OperationalError
|
from sqlalchemy.exc import OperationalError
|
||||||
@ -33,11 +34,9 @@ from sqlalchemy.orm import Session
|
|||||||
from sqlalchemy import PrimaryKeyConstraint
|
from sqlalchemy import PrimaryKeyConstraint
|
||||||
from sqlalchemy.sql.expression import cast
|
from sqlalchemy.sql.expression import cast
|
||||||
from sqlalchemy.sql import select
|
from sqlalchemy.sql import select
|
||||||
from sqlalchemy.types import UserDefinedType, NullType
|
from sqlalchemy.types import UserDefinedType
|
||||||
from sqlalchemy.dialects.postgresql import psycopg2
|
|
||||||
|
|
||||||
from oslo_db import exception
|
from oslo_db import exception
|
||||||
from oslo_db.sqlalchemy.compat import utils as compat_utils
|
|
||||||
from oslo_db.sqlalchemy import models
|
from oslo_db.sqlalchemy import models
|
||||||
from oslo_db.sqlalchemy import provision
|
from oslo_db.sqlalchemy import provision
|
||||||
from oslo_db.sqlalchemy import session
|
from oslo_db.sqlalchemy import session
|
||||||
@ -47,7 +46,6 @@ from oslo_db.tests import utils as test_utils
|
|||||||
|
|
||||||
|
|
||||||
Base = declarative_base()
|
Base = declarative_base()
|
||||||
SA_VERSION = compat_utils.SQLA_VERSION
|
|
||||||
|
|
||||||
|
|
||||||
class TestSanitizeDbUrl(test_base.BaseTestCase):
|
class TestSanitizeDbUrl(test_base.BaseTestCase):
|
||||||
@ -841,12 +839,6 @@ class TestMigrationUtils(db_test_base._DbTestCase):
|
|||||||
Column('deleted', Boolean))
|
Column('deleted', Boolean))
|
||||||
table.create()
|
table.create()
|
||||||
|
|
||||||
# reflection of custom types has been fixed upstream
|
|
||||||
if SA_VERSION < (0, 9, 0):
|
|
||||||
self.assertRaises(exception.ColumnError,
|
|
||||||
utils.change_deleted_column_type_to_id_type,
|
|
||||||
self.engine, table_name)
|
|
||||||
|
|
||||||
fooColumn = Column('foo', CustomType())
|
fooColumn = Column('foo', CustomType())
|
||||||
utils.change_deleted_column_type_to_id_type(self.engine, table_name,
|
utils.change_deleted_column_type_to_id_type(self.engine, table_name,
|
||||||
foo=fooColumn)
|
foo=fooColumn)
|
||||||
@ -908,11 +900,6 @@ class TestMigrationUtils(db_test_base._DbTestCase):
|
|||||||
foo=fooColumn)
|
foo=fooColumn)
|
||||||
|
|
||||||
table = utils.get_table(self.engine, table_name)
|
table = utils.get_table(self.engine, table_name)
|
||||||
# NOTE(boris-42): There is no way to check has foo type CustomType.
|
|
||||||
# but sqlalchemy will set it to NullType. This has
|
|
||||||
# been fixed upstream in recent SA versions
|
|
||||||
if SA_VERSION < (0, 9, 0):
|
|
||||||
self.assertIsInstance(table.c.foo.type, NullType)
|
|
||||||
self.assertIsInstance(table.c.deleted.type, Boolean)
|
self.assertIsInstance(table.c.deleted.type, Boolean)
|
||||||
|
|
||||||
def test_detect_boolean_deleted_constraint_detection(self):
|
def test_detect_boolean_deleted_constraint_detection(self):
|
||||||
|
@ -8,7 +8,7 @@ debtcollector>=1.2.0 # Apache-2.0
|
|||||||
oslo.i18n>=3.15.3 # Apache-2.0
|
oslo.i18n>=3.15.3 # Apache-2.0
|
||||||
oslo.config>=5.2.0 # Apache-2.0
|
oslo.config>=5.2.0 # Apache-2.0
|
||||||
oslo.utils>=3.33.0 # Apache-2.0
|
oslo.utils>=3.33.0 # Apache-2.0
|
||||||
SQLAlchemy!=1.1.5,!=1.1.6,!=1.1.7,!=1.1.8,>=1.0.10 # MIT
|
SQLAlchemy>=1.2.0 # MIT
|
||||||
sqlalchemy-migrate>=0.11.0 # Apache-2.0
|
sqlalchemy-migrate>=0.11.0 # Apache-2.0
|
||||||
stevedore>=1.20.0 # Apache-2.0
|
stevedore>=1.20.0 # Apache-2.0
|
||||||
six>=1.10.0 # MIT
|
six>=1.10.0 # MIT
|
||||||
|
Loading…
Reference in New Issue
Block a user