Merge "Support DB auto-create suppression."

This commit is contained in:
Jenkins 2012-05-21 21:19:30 +00:00 committed by Gerrit Code Review
commit c5a0b0086b
8 changed files with 130 additions and 49 deletions

View File

@ -44,6 +44,7 @@ from glance.common import config
from glance.common import exception from glance.common import exception
from glance.openstack.common import cfg from glance.openstack.common import cfg
import glance.registry.db import glance.registry.db
import glance.registry.db.api
import glance.registry.db.migration import glance.registry.db.migration
@ -75,7 +76,14 @@ def do_version_control(conf, args):
def do_db_sync(conf, args): def do_db_sync(conf, args):
"""Place a database under migration control and upgrade""" """
Place a database under migration control and upgrade,
creating first if necessary.
"""
# override auto-create flag, as complete DB should always
# be created on sync if not already existing
conf.db_auto_create = True
glance.registry.db.api.configure_db(conf)
version = args.pop(0) if args else None version = args.pop(0) if args else None
current_version = args.pop(0) if args else None current_version = args.pop(0) if args else None
glance.registry.db.migration.db_sync(conf, version, current_version) glance.registry.db.migration.db_sync(conf, version, current_version)

View File

@ -68,7 +68,8 @@ db_opts = [
cfg.IntOpt('sql_idle_timeout', default=3600), cfg.IntOpt('sql_idle_timeout', default=3600),
cfg.StrOpt('sql_connection', default='sqlite:///glance.sqlite'), cfg.StrOpt('sql_connection', default='sqlite:///glance.sqlite'),
cfg.IntOpt('sql_max_retries', default=10), cfg.IntOpt('sql_max_retries', default=10),
cfg.IntOpt('sql_retry_interval', default=1) cfg.IntOpt('sql_retry_interval', default=1),
cfg.BoolOpt('db_auto_create', default=True),
] ]
@ -102,7 +103,10 @@ def configure_db(conf):
""" """
global _ENGINE, sa_logger, logger, _MAX_RETRIES, _RETRY_INTERVAL global _ENGINE, sa_logger, logger, _MAX_RETRIES, _RETRY_INTERVAL
if not _ENGINE: if not _ENGINE:
conf.register_opts(db_opts) for opt in db_opts:
# avoid duplicate registration
if not opt.name in conf:
conf.register_opt(opt)
sql_connection = conf.sql_connection sql_connection = conf.sql_connection
_MAX_RETRIES = conf.sql_max_retries _MAX_RETRIES = conf.sql_max_retries
_RETRY_INTERVAL = conf.sql_retry_interval _RETRY_INTERVAL = conf.sql_retry_interval
@ -131,12 +135,16 @@ def configure_db(conf):
elif conf.verbose: elif conf.verbose:
sa_logger.setLevel(logging.INFO) sa_logger.setLevel(logging.INFO)
models.register_models(_ENGINE) if conf.db_auto_create:
try: logger.info('auto-creating glance registry DB')
migration.version_control(conf) models.register_models(_ENGINE)
except exception.DatabaseMigrationError: try:
# only arises when the DB exists and is under version control migration.version_control(conf)
pass except exception.DatabaseMigrationError:
# only arises when the DB exists and is under version control
pass
else:
logger.info('not auto-creating glance registry DB')
def check_mutate_authorization(context, image_ref): def check_mutate_authorization(context, image_ref):

View File

@ -43,32 +43,6 @@ from glance.tests import utils as test_utils
execute, get_unused_port = test_utils.execute, test_utils.get_unused_port execute, get_unused_port = test_utils.execute, test_utils.get_unused_port
def runs_sql(func):
"""
Decorator for a test case method that ensures that the
sql_connection setting is overridden to ensure a disk-based
SQLite database so that arbitrary SQL statements can be
executed out-of-process against the datastore...
"""
@functools.wraps(func)
def wrapped(*a, **kwargs):
test_obj = a[0]
orig_reg_sql_connection = test_obj.registry_server.sql_connection
orig_api_sql_connection = test_obj.api_server.sql_connection
try:
if orig_reg_sql_connection.startswith('sqlite'):
test_obj.registry_server.sql_connection =\
"sqlite:///tests.sqlite"
if orig_api_sql_connection.startswith('sqlite'):
test_obj.api_server.sql_connection =\
"sqlite:///tests.sqlite"
func(*a, **kwargs)
finally:
test_obj.registry_server.sql_connection = orig_reg_sql_connection
test_obj.api_server.sql_connection = orig_api_sql_connection
return wrapped
class Server(object): class Server(object):
""" """
Class used to easily manage starting and stopping Class used to easily manage starting and stopping
@ -94,6 +68,7 @@ class Server(object):
self.exec_env = None self.exec_env = None
self.deployment_flavor = '' self.deployment_flavor = ''
self.server_control_options = '' self.server_control_options = ''
self.needs_database = False
def write_conf(self, **kwargs): def write_conf(self, **kwargs):
""" """
@ -150,6 +125,8 @@ class Server(object):
# Ensure the configuration file is written # Ensure the configuration file is written
overridden = self.write_conf(**kwargs)[1] overridden = self.write_conf(**kwargs)[1]
self.create_database()
cmd = ("%(server_control)s %(server_name)s start " cmd = ("%(server_control)s %(server_name)s start "
"%(conf_file_name)s --pid-file=%(pid_file)s " "%(conf_file_name)s --pid-file=%(pid_file)s "
"%(server_control_options)s" "%(server_control_options)s"
@ -161,6 +138,23 @@ class Server(object):
expected_exitcode=expected_exitcode, expected_exitcode=expected_exitcode,
context=overridden) context=overridden)
def create_database(self):
"""Create database if required for this server"""
if self.needs_database:
conf_dir = os.path.join(self.test_dir, 'etc')
utils.safe_mkdirs(conf_dir)
conf_filepath = os.path.join(conf_dir, 'glance-manage.conf')
with open(conf_filepath, 'wb') as conf_file:
conf_file.write('[DEFAULT]\n')
conf_file.write('sql_connection = %s' % self.sql_connection)
conf_file.flush()
cmd = ('bin/glance-manage db_sync --config-file %s'
% conf_filepath)
execute(cmd, no_venv=self.no_venv, exec_env=self.exec_env,
expect_exit=True)
def stop(self): def stop(self):
""" """
Spin down the server. Spin down the server.
@ -219,7 +213,8 @@ class ApiServer(Server):
self.policy_default_rule = 'default' self.policy_default_rule = 'default'
self.server_control_options = '--capture-output' self.server_control_options = '--capture-output'
default_sql_connection = 'sqlite:///' self.needs_database = True
default_sql_connection = 'sqlite:///tests.sqlite'
self.sql_connection = os.environ.get('GLANCE_TEST_SQL_CONNECTION', self.sql_connection = os.environ.get('GLANCE_TEST_SQL_CONNECTION',
default_sql_connection) default_sql_connection)
@ -260,6 +255,7 @@ image_cache_dir = %(image_cache_dir)s
image_cache_driver = %(image_cache_driver)s image_cache_driver = %(image_cache_driver)s
policy_file = %(policy_file)s policy_file = %(policy_file)s
policy_default_rule = %(policy_default_rule)s policy_default_rule = %(policy_default_rule)s
db_auto_create = False
sql_connection = %(sql_connection)s sql_connection = %(sql_connection)s
[paste_deploy] [paste_deploy]
flavor = %(deployment_flavor)s flavor = %(deployment_flavor)s
@ -338,7 +334,8 @@ class RegistryServer(Server):
super(RegistryServer, self).__init__(test_dir, port) super(RegistryServer, self).__init__(test_dir, port)
self.server_name = 'registry' self.server_name = 'registry'
default_sql_connection = 'sqlite:///' self.needs_database = True
default_sql_connection = 'sqlite:///tests.sqlite'
self.sql_connection = os.environ.get('GLANCE_TEST_SQL_CONNECTION', self.sql_connection = os.environ.get('GLANCE_TEST_SQL_CONNECTION',
default_sql_connection) default_sql_connection)
@ -353,6 +350,7 @@ debug = %(debug)s
bind_host = 0.0.0.0 bind_host = 0.0.0.0
bind_port = %(bind_port)s bind_port = %(bind_port)s
log_file = %(log_file)s log_file = %(log_file)s
db_auto_create = False
sql_connection = %(sql_connection)s sql_connection = %(sql_connection)s
sql_idle_timeout = 3600 sql_idle_timeout = 3600
api_limit_max = 1000 api_limit_max = 1000
@ -671,11 +669,6 @@ class FunctionalTest(unittest.TestCase):
if os.path.exists(self.test_dir): if os.path.exists(self.test_dir):
shutil.rmtree(self.test_dir) shutil.rmtree(self.test_dir)
# We do this here because the @runs_sql decorator above
# actually resets the registry server's sql_connection
# to the original (usually memory-based SQLite connection)
# and this block of code is run *before* the finally:
# block in that decorator...
self._reset_database(self.registry_server.sql_connection) self._reset_database(self.registry_server.sql_connection)
def run_sql_cmd(self, sql): def run_sql_cmd(self, sql):

View File

@ -579,7 +579,6 @@ class TestBinGlance(functional.FunctionalTest):
self.stop_servers() self.stop_servers()
@functional.runs_sql
def test_add_location_with_checksum(self): def test_add_location_with_checksum(self):
""" """
We test the following: We test the following:
@ -611,7 +610,6 @@ class TestBinGlance(functional.FunctionalTest):
self.stop_servers() self.stop_servers()
@functional.runs_sql
def test_add_location_without_checksum(self): def test_add_location_without_checksum(self):
""" """
We test the following: We test the following:
@ -643,7 +641,6 @@ class TestBinGlance(functional.FunctionalTest):
self.stop_servers() self.stop_servers()
@functional.runs_sql
def test_add_clear(self): def test_add_clear(self):
""" """
We test the following: We test the following:

View File

@ -0,0 +1,77 @@
# vim: tabstop=4 shiftwidth=4 softtabstop=4
# Copyright 2012 Red Hat, Inc
# All Rights Reserved.
#
# 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.
"""Functional test cases for glance-manage"""
import os
from glance.common import utils
from glance.tests import functional
from glance.tests.utils import execute, depends_on_exe, skip_if_disabled
class TestGlanceManage(functional.FunctionalTest):
"""Functional tests for glance-manage"""
def setUp(self):
super(TestGlanceManage, self).setUp()
conf_dir = os.path.join(self.test_dir, 'etc')
utils.safe_mkdirs(conf_dir)
self.conf_filepath = os.path.join(conf_dir, 'glance-manage.conf')
self.db_filepath = os.path.join(conf_dir, 'test.sqlite')
self.connection = ('sql_connection = sqlite:///%s' %
self.db_filepath)
def _sync_db(self, auto_create):
with open(self.conf_filepath, 'wb') as conf_file:
conf_file.write('[DEFAULT]\n')
conf_file.write('db_auto_create = %r\n' % auto_create)
conf_file.write(self.connection)
conf_file.flush()
cmd = ('bin/glance-manage db_sync --config-file %s' %
self.conf_filepath)
execute(cmd, raise_error=True)
def _assert_tables(self):
cmd = "sqlite3 %s '.schema'" % self.db_filepath
exitcode, out, err = execute(cmd, raise_error=True)
self.assertTrue('CREATE TABLE images' in out)
self.assertTrue('CREATE TABLE image_tags' in out)
self.assertTrue('CREATE TABLE image_members' in out)
self.assertTrue('CREATE TABLE image_properties' in out)
@depends_on_exe('sqlite3')
@skip_if_disabled
def test_db_creation(self):
"""Test DB creation by db_sync on a fresh DB"""
self._sync_db(True)
self._assert_tables()
self.stop_servers()
@depends_on_exe('sqlite3')
@skip_if_disabled
def test_db_creation_auto_create_overridden(self):
"""Test DB creation with db_auto_create False"""
self._sync_db(False)
self._assert_tables()
self.stop_servers()

View File

@ -25,7 +25,6 @@ from glance.tests.utils import execute, depends_on_exe, skip_if_disabled
class TestSqlite(functional.FunctionalTest): class TestSqlite(functional.FunctionalTest):
"""Functional tests for sqlite-specific logic""" """Functional tests for sqlite-specific logic"""
@functional.runs_sql
@depends_on_exe('sqlite3') @depends_on_exe('sqlite3')
@skip_if_disabled @skip_if_disabled
def test_big_int_mapping(self): def test_big_int_mapping(self):

View File

@ -1353,3 +1353,5 @@ class TestApi(functional.FunctionalTest):
response, content = http.request(path, 'HEAD', headers=auth_headers) response, content = http.request(path, 'HEAD', headers=auth_headers)
self.assertEqual(response.status, 200) self.assertEqual(response.status, 200)
self.assertEqual('tenant2', response['x-image-meta-owner']) self.assertEqual('tenant2', response['x-image-meta-owner'])
self.stop_servers()

View File

@ -51,7 +51,6 @@ class TestImages(functional.FunctionalTest):
base_headers.update(custom_headers or {}) base_headers.update(custom_headers or {})
return base_headers return base_headers
@functional.runs_sql
def test_image_lifecycle(self): def test_image_lifecycle(self):
# Image list should be empty # Image list should be empty
path = self._url('/images') path = self._url('/images')
@ -181,7 +180,6 @@ class TestImages(functional.FunctionalTest):
self.stop_servers() self.stop_servers()
@functional.runs_sql
def test_permissions(self): def test_permissions(self):
# Create an image that belongs to TENANT1 # Create an image that belongs to TENANT1
path = self._url('/images') path = self._url('/images')
@ -410,7 +408,6 @@ class TestImages(functional.FunctionalTest):
self.stop_servers() self.stop_servers()
@functional.runs_sql
def test_tag_lifecycle(self): def test_tag_lifecycle(self):
# Create an image for our tests # Create an image for our tests
path = self._url('/images') path = self._url('/images')