Added Postgres CI opportunistic test case
Added test case and support functions for CI opportunistic testing using a PostgreSQL database, as part of our postgresql as a first class citizen effort. rebased again on top of test migration with data framework clean up debug, and ensure pg will fail to connect gracefully fix previous migration test which was inserting a key by id, which postgresql doesn't like, as it doesn't adjust the autoincrement counter. added driver to connection string for mysql Updated comment to expose how to do this on pg easily Fixes: bug 1084567 Change-Id: If0bfe5cf748f6d5724d17082be6d18b6f96f6ee4
This commit is contained in:
parent
38c7fa2ab9
commit
5dcf4dd9b2
@ -22,6 +22,22 @@ to use in the tests. For each connection found in the config file,
|
|||||||
the test case runs a series of test cases to ensure that migrations work
|
the test case runs a series of test cases to ensure that migrations work
|
||||||
properly both upgrading and downgrading, and that no data loss occurs
|
properly both upgrading and downgrading, and that no data loss occurs
|
||||||
if possible.
|
if possible.
|
||||||
|
|
||||||
|
There are also "opportunistic" tests for both mysql and postgresql in here,
|
||||||
|
which allows testing against all 3 databases (sqlite in memory, mysql, pg) in
|
||||||
|
a properly configured unit test environment.
|
||||||
|
|
||||||
|
For the opportunistic testing you need to set up a db named 'openstack_citest'
|
||||||
|
with user 'openstack_citest' and password 'openstack_citest' on localhost.
|
||||||
|
The test will then use that db and u/p combo to run the tests.
|
||||||
|
|
||||||
|
For postgres on Ubuntu this can be done with the following commands:
|
||||||
|
|
||||||
|
sudo -u postgres psql
|
||||||
|
postgres=# create user openstack_citest with createdb login password
|
||||||
|
'openstack_citest';
|
||||||
|
postgres=# create database openstack_citest with owner openstack_citest;
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
import collections
|
import collections
|
||||||
@ -53,6 +69,8 @@ def _get_connect_string(backend,
|
|||||||
"""
|
"""
|
||||||
if backend == "postgres":
|
if backend == "postgres":
|
||||||
backend = "postgresql+psycopg2"
|
backend = "postgresql+psycopg2"
|
||||||
|
elif backend == "mysql":
|
||||||
|
backend = "mysql+mysqldb"
|
||||||
|
|
||||||
return ("%(backend)s://%(user)s:%(passwd)s@localhost/%(database)s"
|
return ("%(backend)s://%(user)s:%(passwd)s@localhost/%(database)s"
|
||||||
% locals())
|
% locals())
|
||||||
@ -194,11 +212,12 @@ class TestMigrations(test.TestCase):
|
|||||||
password = ""
|
password = ""
|
||||||
if len(auth_pieces) > 1:
|
if len(auth_pieces) > 1:
|
||||||
password = auth_pieces[1].strip()
|
password = auth_pieces[1].strip()
|
||||||
# note(boris-42): This file is used for authentication
|
# note(krtaylor): File creation problems with tests in
|
||||||
# without password prompt.
|
# venv using .pgpass authentication, changed to
|
||||||
createpgpass = ("echo '*:*:*:%(user)s:%(password)s' > "
|
# PGPASSWORD environment variable which is no longer
|
||||||
"~/.pgpass && chmod 0600 ~/.pgpass" % locals())
|
# planned to be deprecated
|
||||||
execute_cmd(createpgpass)
|
os.environ['PGPASSWORD'] = password
|
||||||
|
os.environ['PGUSER'] = user
|
||||||
# note(boris-42): We must create and drop database, we can't
|
# note(boris-42): We must create and drop database, we can't
|
||||||
# drop database which we have connected to, so for such
|
# drop database which we have connected to, so for such
|
||||||
# operations there is a special database template1.
|
# operations there is a special database template1.
|
||||||
@ -210,6 +229,8 @@ class TestMigrations(test.TestCase):
|
|||||||
sql = ("create database %(database)s;") % locals()
|
sql = ("create database %(database)s;") % locals()
|
||||||
createtable = sqlcmd % locals()
|
createtable = sqlcmd % locals()
|
||||||
execute_cmd(createtable)
|
execute_cmd(createtable)
|
||||||
|
os.unsetenv('PGPASSWORD')
|
||||||
|
os.unsetenv('PGUSER')
|
||||||
|
|
||||||
def test_walk_versions(self):
|
def test_walk_versions(self):
|
||||||
"""
|
"""
|
||||||
@ -258,6 +279,29 @@ class TestMigrations(test.TestCase):
|
|||||||
self.assertEqual(count, 0, "%d non InnoDB tables created" % count)
|
self.assertEqual(count, 0, "%d non InnoDB tables created" % count)
|
||||||
connection.close()
|
connection.close()
|
||||||
|
|
||||||
|
def test_postgresql_connect_fail(self):
|
||||||
|
"""
|
||||||
|
Test that we can trigger a postgres connection failure and we fail
|
||||||
|
gracefully to ensure we don't break people without postgres
|
||||||
|
"""
|
||||||
|
if _is_backend_avail('postgresql', user="openstack_cifail"):
|
||||||
|
self.fail("Shouldn't have connected")
|
||||||
|
|
||||||
|
def test_postgresql_opportunistically(self):
|
||||||
|
# Test postgresql database migration walk
|
||||||
|
if not _is_backend_avail('postgres'):
|
||||||
|
self.skipTest("postgresql not available")
|
||||||
|
# add this to the global lists to make reset work with it, it's removed
|
||||||
|
# automatically in tearDown so no need to clean it up here.
|
||||||
|
connect_string = _get_connect_string("postgres")
|
||||||
|
engine = sqlalchemy.create_engine(connect_string)
|
||||||
|
self.engines["postgresqlcitest"] = engine
|
||||||
|
self.test_databases["postgresqlcitest"] = connect_string
|
||||||
|
|
||||||
|
# build a fully populated postgresql database with all the tables
|
||||||
|
self._reset_databases()
|
||||||
|
self._walk_versions(engine, False, False)
|
||||||
|
|
||||||
def _walk_versions(self, engine=None, snake_walk=False, downgrade=True):
|
def _walk_versions(self, engine=None, snake_walk=False, downgrade=True):
|
||||||
# Determine latest version script from the repo, then
|
# Determine latest version script from the repo, then
|
||||||
# upgrade from 1 through to the latest, with no data
|
# upgrade from 1 through to the latest, with no data
|
||||||
@ -311,6 +355,9 @@ class TestMigrations(test.TestCase):
|
|||||||
migration version with special _prerun_### and
|
migration version with special _prerun_### and
|
||||||
_check_### functions in the main test.
|
_check_### functions in the main test.
|
||||||
"""
|
"""
|
||||||
|
# NOTE(sdague): try block is here because it's impossible to debug
|
||||||
|
# where a failed data migration happens otherwise
|
||||||
|
try:
|
||||||
if with_data:
|
if with_data:
|
||||||
data = None
|
data = None
|
||||||
prerun = getattr(self, "_prerun_%d" % version, None)
|
prerun = getattr(self, "_prerun_%d" % version, None)
|
||||||
@ -320,7 +367,8 @@ class TestMigrations(test.TestCase):
|
|||||||
migration_api.upgrade(engine,
|
migration_api.upgrade(engine,
|
||||||
TestMigrations.REPOSITORY,
|
TestMigrations.REPOSITORY,
|
||||||
version)
|
version)
|
||||||
self.assertEqual(version,
|
self.assertEqual(
|
||||||
|
version,
|
||||||
migration_api.db_version(engine,
|
migration_api.db_version(engine,
|
||||||
TestMigrations.REPOSITORY))
|
TestMigrations.REPOSITORY))
|
||||||
|
|
||||||
@ -328,24 +376,30 @@ class TestMigrations(test.TestCase):
|
|||||||
check = getattr(self, "_check_%d" % version, None)
|
check = getattr(self, "_check_%d" % version, None)
|
||||||
if check:
|
if check:
|
||||||
check(engine, data)
|
check(engine, data)
|
||||||
|
except Exception:
|
||||||
|
LOG.error("Failed to migrate to version %s on engine %s" %
|
||||||
|
(version, engine))
|
||||||
|
raise
|
||||||
|
|
||||||
# migration 146, availability zone transition
|
# migration 146, availability zone transition
|
||||||
def _prerun_146(self, engine):
|
def _prerun_146(self, engine):
|
||||||
data = {
|
data = {
|
||||||
'id': 1,
|
|
||||||
'availability_zone': 'custom_az',
|
'availability_zone': 'custom_az',
|
||||||
'aggregate_name': 1,
|
'aggregate_name': 1,
|
||||||
'name': 'name',
|
'name': 'name',
|
||||||
}
|
}
|
||||||
|
|
||||||
aggregates = get_table(engine, 'aggregates')
|
aggregates = get_table(engine, 'aggregates')
|
||||||
aggregates.insert().values(data).execute()
|
result = aggregates.insert().values(data).execute()
|
||||||
|
# NOTE(sdague) it's important you don't insert keys by value in
|
||||||
|
# postgresql, because it's autoincrement counter won't get updated
|
||||||
|
data['id'] = result.inserted_primary_key[0]
|
||||||
return data
|
return data
|
||||||
|
|
||||||
def _check_146(self, engine, data):
|
def _check_146(self, engine, data):
|
||||||
aggregate_md = get_table(engine, 'aggregate_metadata')
|
aggregate_md = get_table(engine, 'aggregate_metadata')
|
||||||
md = aggregate_md.select(
|
md = aggregate_md.select(
|
||||||
aggregate_md.c.aggregate_id == 1).execute().first()
|
aggregate_md.c.aggregate_id == data['id']).execute().first()
|
||||||
self.assertEqual(data['availability_zone'], md['value'])
|
self.assertEqual(data['availability_zone'], md['value'])
|
||||||
|
|
||||||
# migration 147, availability zone transition for services
|
# migration 147, availability zone transition for services
|
||||||
|
@ -7,6 +7,7 @@ feedparser
|
|||||||
fixtures>=0.3.12
|
fixtures>=0.3.12
|
||||||
mox==0.5.3
|
mox==0.5.3
|
||||||
MySQL-python
|
MySQL-python
|
||||||
|
psycopg2
|
||||||
pep8==1.3.3
|
pep8==1.3.3
|
||||||
pylint==0.25.2
|
pylint==0.25.2
|
||||||
python-subunit
|
python-subunit
|
||||||
|
Loading…
Reference in New Issue
Block a user