From 1f3a4fda8712dadaa4f2fe743ac566907d31d3c1 Mon Sep 17 00:00:00 2001 From: Craig Vyvial Date: Thu, 23 May 2013 10:33:58 -0700 Subject: [PATCH] adding configuration group support This allows users to make custom configurations groups and apply them to multiple instances. Configuration parameters that can be set are listed in the validation_rules.json file. implements blueprint configuration-management Change-Id: I99b0bfe51c131ace30774e75e30d620101ed2e0b --- trove/common/api.py | 64 ++ trove/common/apischema.py | 84 ++- trove/common/cfg.py | 6 + trove/common/configurations.py | 84 +++ trove/common/exception.py | 29 + trove/common/template.py | 43 +- trove/common/utils.py | 7 + trove/common/wsgi.py | 6 + trove/configuration/__init__.py | 0 trove/configuration/models.py | 215 +++++++ trove/configuration/service.py | 287 +++++++++ trove/configuration/views.py | 107 ++++ trove/db/sqlalchemy/mappers.py | 4 + .../versions/020_configurations.py | 65 ++ trove/db/sqlalchemy/session.py | 2 + trove/guestagent/api.py | 16 +- trove/guestagent/common/sql_query.py | 19 + trove/guestagent/datastore/mysql/manager.py | 12 +- trove/guestagent/datastore/mysql/service.py | 70 +- trove/instance/models.py | 118 +++- trove/instance/service.py | 48 +- trove/instance/tasks.py | 2 + trove/instance/views.py | 22 + trove/taskmanager/api.py | 27 +- trove/taskmanager/manager.py | 14 +- trove/taskmanager/models.py | 113 +++- trove/templates/mysql/config.template | 6 +- .../templates/mysql/override.config.template | 12 + trove/templates/mysql/validation-rules.json | 224 +++++++ trove/tests/api/configurations.py | 606 ++++++++++++++++++ trove/tests/api/instances.py | 1 + trove/tests/fakes/guestagent.py | 10 +- .../tests/unittests/configuration/__init__.py | 0 .../test_configuration_controller.py | 102 +++ trove/tests/unittests/datastore/__init__.py | 0 .../unittests/datastore/test_datastore.py | 28 + trove/tests/unittests/guestagent/test_api.py | 26 +- .../tests/unittests/guestagent/test_dbaas.py | 15 +- .../unittests/guestagent/test_manager.py | 8 +- .../unittests/instance/test_instance_views.py | 3 + trove/tests/util/__init__.py | 1 + 41 files changed, 2439 insertions(+), 67 deletions(-) create mode 100644 trove/common/configurations.py create mode 100644 trove/configuration/__init__.py create mode 100644 trove/configuration/models.py create mode 100644 trove/configuration/service.py create mode 100644 trove/configuration/views.py create mode 100644 trove/db/sqlalchemy/migrate_repo/versions/020_configurations.py create mode 100644 trove/templates/mysql/override.config.template create mode 100644 trove/templates/mysql/validation-rules.json create mode 100644 trove/tests/api/configurations.py create mode 100644 trove/tests/unittests/configuration/__init__.py create mode 100644 trove/tests/unittests/configuration/test_configuration_controller.py create mode 100644 trove/tests/unittests/datastore/__init__.py create mode 100644 trove/tests/unittests/datastore/test_datastore.py diff --git a/trove/common/api.py b/trove/common/api.py index 8206abb6c2..65e567a57e 100644 --- a/trove/common/api.py +++ b/trove/common/api.py @@ -15,6 +15,8 @@ import routes from trove.common import wsgi +from trove.configuration.service import ConfigurationsController +from trove.configuration.service import ParametersController from trove.flavor.service import FlavorController from trove.instance.service import InstanceController from trove.limits.service import LimitsController @@ -34,6 +36,7 @@ class API(wsgi.Router): self._versions_router(mapper) self._limits_router(mapper) self._backups_router(mapper) + self._configurations_router(mapper) def _versions_router(self, mapper): versions_resource = VersionsController().create_resource() @@ -74,6 +77,10 @@ class API(wsgi.Router): controller=instance_resource, action="action", conditions={'method': ['POST']}) + mapper.connect("/{tenant_id}/instances/{id}", + controller=instance_resource, + action="update", + conditions={'method': ['PUT']}) mapper.connect("/{tenant_id}/instances/{id}", controller=instance_resource, action="delete", @@ -82,6 +89,10 @@ class API(wsgi.Router): controller=instance_resource, action="backups", conditions={'method': ['GET']}) + mapper.connect("/{tenant_id}/instances/{id}/configuration", + controller=instance_resource, + action="configuration", + conditions={'method': ['GET']}) def _flavor_router(self, mapper): flavor_resource = FlavorController().create_resource() @@ -124,6 +135,59 @@ class API(wsgi.Router): action="delete", conditions={'method': ['DELETE']}) + def _configurations_router(self, mapper): + parameters_resource = ParametersController().create_resource() + path = '/{tenant_id}/datastores/versions/{version}/parameters' + mapper.connect(path, + controller=parameters_resource, + action='index_by_version', + conditions={'method': ['GET']}) + path = '/{tenant_id}/datastores/versions/{version}/parameters/{name}' + mapper.connect(path, + controller=parameters_resource, + action='show_by_version', + conditions={'method': ['GET']}) + + path = '/{tenant_id}/datastores/{datastore}/versions/{id}' + mapper.connect(path + '/parameters', + controller=parameters_resource, + action='index', + conditions={'method': ['GET']}) + mapper.connect(path + '/parameters/{name}', + controller=parameters_resource, + action='show', + conditions={'method': ['GET']}) + + configuration_resource = ConfigurationsController().create_resource() + mapper.connect('/{tenant_id}/configurations', + controller=configuration_resource, + action='index', + conditions={'method': ['GET']}) + mapper.connect('/{tenant_id}/configurations', + controller=configuration_resource, + action='create', + conditions={'method': ['POST']}) + mapper.connect('/{tenant_id}/configurations/{id}', + controller=configuration_resource, + action='show', + conditions={'method': ['GET']}) + mapper.connect('/{tenant_id}/configurations/{id}/instances', + controller=configuration_resource, + action='instances', + conditions={'method': ['GET']}) + mapper.connect('/{tenant_id}/configurations/{id}', + controller=configuration_resource, + action='edit', + conditions={'method': ['PATCH']}) + mapper.connect('/{tenant_id}/configurations/{id}', + controller=configuration_resource, + action='update', + conditions={'method': ['PUT']}) + mapper.connect('/{tenant_id}/configurations/{id}', + controller=configuration_resource, + action='delete', + conditions={'method': ['DELETE']}) + def app_factory(global_conf, **local_conf): return API() diff --git a/trove/common/apischema.py b/trove/common/apischema.py index 2947a44bf8..ef0ef895c6 100644 --- a/trove/common/apischema.py +++ b/trove/common/apischema.py @@ -17,14 +17,16 @@ from trove.common import cfg CONF = cfg.CONF +url_ref = { + "type": "string", + "minLength": 8, + "pattern": 'http[s]?://(?:[a-zA-Z]|[0-9]|[$-_@.&+]|[!*\(\),]' + '|(?:%[0-9a-fA-F][0-9a-fA-F]))+' +} + flavorref = { 'oneOf': [ - { - "type": "string", - "minLength": 8, - "pattern": 'http[s]?://(?:[a-zA-Z]|[0-9]|[$-_@.&+]|[!*\(\),]' - '|(?:%[0-9a-fA-F][0-9a-fA-F]))+' - }, + url_ref, { "type": "string", "maxLength": 5, @@ -172,6 +174,12 @@ users_list = { } } +configuration_id = { + 'oneOf': [ + uuid + ] +} + instance = { "create": { "type": "object", @@ -185,6 +193,7 @@ instance = { "additionalProperties": True, "properties": { "name": non_empty_string, + "configuration_id": configuration_id, "flavorRef": flavorref, "volume": volume, "databases": databases_def, @@ -352,6 +361,69 @@ backup = { } } +configuration = { + "create": { + "name": "configuration:create", + "type": "object", + "required": ["configuration"], + "properties": { + "configuration": { + "type": "object", + "required": ["values", "name"], + "properties": { + "description": non_empty_string, + "values": { + "type": "object", + }, + "name": non_empty_string, + "datastore": { + "type": "object", + "additionalProperties": True, + "properties": { + "type": non_empty_string, + "version": non_empty_string + } + } + } + } + } + }, + "update": { + "name": "configuration:update", + "type": "object", + "required": ["configuration"], + "properties": { + "configuration": { + "type": "object", + "required": [], + "properties": { + "description": non_empty_string, + "values": { + "type": "object", + }, + "name": non_empty_string + } + } + } + }, + "edit": { + "name": "configuration:edit", + "type": "object", + "required": ["configuration"], + "properties": { + "configuration": { + "type": "object", + "required": [], + "properties": { + "values": { + "type": "object", + } + } + } + } + } +} + account = { 'create': { "type": "object", diff --git a/trove/common/cfg.py b/trove/common/cfg.py index e0d2a60959..1a9688e1e7 100644 --- a/trove/common/cfg.py +++ b/trove/common/cfg.py @@ -1,6 +1,7 @@ # vim: tabstop=4 shiftwidth=4 softtabstop=4 # Copyright 2011 OpenStack Foundation +# Copyright 2014 Rackspace Hosting # All Rights Reserved. # # Licensed under the Apache License, Version 2.0 (the "License"); you may @@ -17,6 +18,8 @@ """Routines for configuring Trove.""" from oslo.config import cfg +from trove.openstack.common import log as logging + import os.path UNKNOWN_SERVICE_ID = 'unknown-service-id-error' @@ -28,6 +31,8 @@ path_opts = [ help='Directory where the trove python module is installed.'), ] +LOG = logging.getLogger(__name__) + common_opts = [ cfg.StrOpt('sql_connection', default='sqlite:///trove_test.sqlite', @@ -80,6 +85,7 @@ common_opts = [ cfg.IntOpt('databases_page_size', default=20), cfg.IntOpt('instances_page_size', default=20), cfg.IntOpt('backups_page_size', default=20), + cfg.IntOpt('configurations_page_size', default=20), cfg.ListOpt('ignore_users', default=['os_admin', 'root']), cfg.ListOpt('ignore_dbs', default=['lost+found', 'mysql', diff --git a/trove/common/configurations.py b/trove/common/configurations.py new file mode 100644 index 0000000000..ea2b28f7ef --- /dev/null +++ b/trove/common/configurations.py @@ -0,0 +1,84 @@ +# Copyright 2014 Rackspace Hosting +# 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. + +import io +import json +from trove.common import cfg +from trove.common import exception +from trove.common import utils +from trove.openstack.common import log as logging +from trove.openstack.common.gettextutils import _ +from six.moves import configparser + + +LOG = logging.getLogger(__name__) +CONF = cfg.CONF +ENV = utils.ENV + + +def _get_item(key, dictList): + for item in dictList: + if key == item.get('name'): + return item + + +def do_configs_require_restart(overrides, datastore_manager='mysql'): + rules = get_validation_rules(datastore_manager=datastore_manager) + LOG.debug(_("overrides: %s") % overrides) + LOG.debug(_("rules?: %s") % rules) + for key in overrides.keys(): + rule = _get_item(key, rules['configuration-parameters']) + LOG.debug(_("checking the rule: %s") % rule) + if rule.get('restart_required'): + return True + return False + + +def get_validation_rules(datastore_manager='mysql'): + try: + config_location = ("%s/validation-rules.json" % datastore_manager) + template = ENV.get_template(config_location) + return json.loads(template.render()) + except Exception: + msg = "This operation is not supported for this datastore at this time" + LOG.exception(msg) + raise exception.UnprocessableEntity(message=msg) + + +class MySQLConfParser(object): + """MySQLConfParser""" + def __init__(self, config): + self.config = config + + def parse(self): + good_cfg = self._remove_commented_lines(str(self.config)) + cfg_parser = configparser.ConfigParser() + cfg_parser.readfp(io.BytesIO(str(good_cfg))) + return cfg_parser.items("mysqld") + + def _remove_commented_lines(self, config_str): + ret = [] + for line in config_str.splitlines(): + line_clean = line.strip() + if line_clean.startswith('#'): + continue + elif line_clean.startswith('!'): + continue + elif line_clean.startswith(':'): + continue + else: + ret.append(line_clean) + rendered = "\n".join(ret) + return rendered diff --git a/trove/common/exception.py b/trove/common/exception.py index 1e9a374b88..9dcad5aec2 100644 --- a/trove/common/exception.py +++ b/trove/common/exception.py @@ -184,6 +184,14 @@ class UserAlreadyExists(BadRequest): message = _('A user with the name "%(name)s" already exists.') +class InstanceAssignedToConfiguration(BadRequest): + + message = _('A configuration group cannot be deleted if it is ' + 'associated with one or more non-terminated instances. ' + 'Detach the configuration group from all non-terminated ' + 'instances and please try again.') + + class UnprocessableEntity(TroveError): message = _("Unable to process the contained request") @@ -357,3 +365,24 @@ class RestoreBackupIntegrityError(TroveError): message = _("Current Swift object checksum does not match original " "checksum for backup %(backup_id)s.") + + +class ConfigKeyNotFound(NotFound): + message = _("%(key)s is not a supported configuration parameter") + + +class NoConfigParserFound(NotFound): + message = _("No configuration parser found for datastore " + "%(datastore_manager)s") + + +class ConfigurationDatastoreNotMatchInstance(TroveError): + message = _("Datastore Version on Configuration " + "%(config_datastore_version)s does not " + "match the Datastore Version on the instance " + "%(instance_datastore_version)s.") + + +class ConfigurationParameterDeleted(object): + message = _("%(parameter_name)s parameter can no longer be " + " set as of %(parameter_deleted_at)s") diff --git a/trove/common/template.py b/trove/common/template.py index 79475f8601..8ba5a67eab 100644 --- a/trove/common/template.py +++ b/trove/common/template.py @@ -1,4 +1,6 @@ # Copyright 2012 OpenStack Foundation +# Copyright 2014 Rackspace Hosting +# 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 @@ -14,16 +16,20 @@ import jinja2 from trove.common import cfg +from trove.common import configurations from trove.common import exception +from trove.common import utils from trove.openstack.common import log as logging CONF = cfg.CONF LOG = logging.getLogger(__name__) -ENV = jinja2.Environment(loader=jinja2.ChoiceLoader([ - jinja2.FileSystemLoader(CONF.template_path), - jinja2.PackageLoader("trove", "templates"), -])) +ENV = utils.ENV + +# TODO(cp16net) Maybe this should be moved to a config dict +SERVICE_PARSERS = { + 'mysql': configurations.MySQLConfParser, +} class SingleInstanceConfigTemplate(object): @@ -31,6 +37,8 @@ class SingleInstanceConfigTemplate(object): rendering on the guest """ + template_name = "%s/config.template" + def __init__(self, datastore_manager, flavor_dict, instance_id): """Constructor @@ -43,21 +51,36 @@ class SingleInstanceConfigTemplate(object): """ self.flavor_dict = flavor_dict - template_filename = "%s/config.template" % datastore_manager + template_filename = self.template_name % datastore_manager self.template = ENV.get_template(template_filename) + self.datastore_manager = datastore_manager self.instance_id = instance_id - def render(self): + def render(self, **kwargs): """Renders the jinja template :returns: str -- The rendered configuration file """ + template = ENV.get_template(self.template_name % + self.datastore_manager) server_id = self._calculate_unique_id() - self.config_contents = self.template.render( - flavor=self.flavor_dict, server_id=server_id) + self.config_contents = template.render( + flavor=self.flavor_dict, server_id=server_id, **kwargs) return self.config_contents + def render_dict(self): + """ + Renders the default configuration template file as a dictionary + to apply the default configuration dynamically. + """ + config = self.render() + cfg_parser = SERVICE_PARSERS.get(self.datastore_manager) + if not cfg_parser: + raise exception.NoConfigParserFound( + datastore_manager=self.datastore_manager) + return cfg_parser(config).parse() + def _calculate_unique_id(self): """ Returns a positive unique id based off of the instance id @@ -67,6 +90,10 @@ class SingleInstanceConfigTemplate(object): return abs(hash(self.instance_id) % (2 ** 31)) +class OverrideConfigTemplate(SingleInstanceConfigTemplate): + template_name = "%s/override.config.template" + + def load_heat_template(datastore_manager): template_filename = "%s/heat.template" % datastore_manager try: diff --git a/trove/common/utils.py b/trove/common/utils.py index cf0fba8714..3db2a3ba40 100644 --- a/trove/common/utils.py +++ b/trove/common/utils.py @@ -18,6 +18,7 @@ import datetime import inspect +import jinja2 import sys import time import urlparse @@ -48,6 +49,12 @@ bool_from_string = openstack_utils.bool_from_string execute = processutils.execute isotime = timeutils.isotime +CONF = cfg.CONF +ENV = jinja2.Environment(loader=jinja2.ChoiceLoader([ + jinja2.FileSystemLoader(CONF.template_path), + jinja2.PackageLoader("trove", "templates") + ])) + def create_method_args_string(*args, **kwargs): """Returns a string representation of args and keyword args. diff --git a/trove/common/wsgi.py b/trove/common/wsgi.py index 16ab4ba334..5868c8683e 100644 --- a/trove/common/wsgi.py +++ b/trove/common/wsgi.py @@ -82,6 +82,12 @@ CUSTOM_SERIALIZER_METADATA = { #mgmt/instance 'id': '', }, + 'configuration': { + 'id': '', + 'name': '', + 'description': '', + 'datastore_version_id': '' + }, 'flavor': {'id': '', 'ram': '', 'name': ''}, 'link': {'href': '', 'rel': ''}, 'database': {'name': ''}, diff --git a/trove/configuration/__init__.py b/trove/configuration/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/trove/configuration/models.py b/trove/configuration/models.py new file mode 100644 index 0000000000..7fa60a3516 --- /dev/null +++ b/trove/configuration/models.py @@ -0,0 +1,215 @@ +# Copyright 2014 Rackspace +# 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. + +from datetime import datetime + +from trove.common import cfg +from trove.common import configurations +from trove.common.exception import ModelNotFoundError +from trove.datastore.models import DatastoreVersion +from trove.db import models as dbmodels +from trove.openstack.common import log as logging +from trove.openstack.common.gettextutils import _ +from trove.taskmanager import api as task_api + + +CONF = cfg.CONF +LOG = logging.getLogger(__name__) + + +class Configurations(object): + + DEFAULT_LIMIT = CONF.configurations_page_size + + @staticmethod + def load(context): + if context is None: + raise TypeError("Argument context not defined.") + elif id is None: + raise TypeError("Argument is not defined.") + + if context.is_admin: + db_info = DBConfiguration.find_all(deleted=False) + if db_info is None: + LOG.debug(_("No configurations found")) + else: + db_info = DBConfiguration.find_all(tenant_id=context.tenant, + deleted=False) + if db_info is None: + LOG.debug(_("No configurations found for tenant % s") + % context.tenant) + + limit = int(context.limit or Configurations.DEFAULT_LIMIT) + if limit > Configurations.DEFAULT_LIMIT: + limit = Configurations.DEFAULT_LIMIT + + data_view = DBConfiguration.find_by_pagination('configurations', + db_info, + "foo", + limit=limit, + marker=context.marker) + next_marker = data_view.next_page_marker + return data_view.collection, next_marker + + +class Configuration(object): + + @property + def instances(self): + return self.instances + + @property + def items(self): + return self.items + + @staticmethod + def create(name, description, tenant_id, datastore, datastore_version): + configurationGroup = DBConfiguration.create( + name=name, + description=description, + tenant_id=tenant_id, + datastore_version_id=datastore_version) + return configurationGroup + + @staticmethod + def create_items(cfg_id, values): + LOG.debug(_("saving the values to the database")) + LOG.debug(_("cfg_id: %s") % cfg_id) + LOG.debug(_("values: %s") % values) + config_items = [] + for key, val in values.iteritems(): + config_item = ConfigurationParameter.create( + configuration_id=cfg_id, + configuration_key=key, + configuration_value=val) + config_items.append(config_item) + return config_items + + @staticmethod + def delete(context, group): + deleted_at = datetime.utcnow() + Configuration.remove_all_items(context, group.id, deleted_at) + group.deleted = True + group.deleted_at = deleted_at + group.save() + + @staticmethod + def remove_all_items(context, id, deleted_at): + LOG.debug(_("removing the values from the database with configuration" + " %s") % id) + items = ConfigurationParameter.find_all(configuration_id=id, + deleted=False).all() + LOG.debug(_("removing items: %s") % items) + for item in items: + item.deleted = True + item.deleted_at = deleted_at + item.save() + + @staticmethod + def load_configuration_datastore_version(context, id): + config = Configuration.load(context, id) + datastore_version = DatastoreVersion.load_by_uuid( + config.datastore_version_id) + return datastore_version + + @staticmethod + def load(context, id): + try: + if context.is_admin: + config_info = DBConfiguration.find_by(id=id, + deleted=False) + else: + config_info = DBConfiguration.find_by(id=id, + tenant_id=context.tenant, + deleted=False) + except ModelNotFoundError: + msg = _("Configuration group with ID %s could not be found.") % id + raise ModelNotFoundError(msg) + return config_info + + @staticmethod + def load_items(context, id): + datastore = Configuration.load_configuration_datastore_version(context, + id) + config_items = ConfigurationParameter.find_all(configuration_id=id, + deleted=False).all() + rules = configurations.get_validation_rules( + datastore_manager=datastore.manager) + + def _get_rule(key): + LOG.debug(_("finding rule with key : %s") % key) + for rule in rules['configuration-parameters']: + if str(rule.get('name')) == key: + return rule + + for item in config_items: + rule = _get_rule(str(item.configuration_key)) + if rule.get('type') == 'boolean': + item.configuration_value = bool(int(item.configuration_value)) + elif rule.get('type') == 'integer': + item.configuration_value = int(item.configuration_value) + else: + item.configuration_value = str(item.configuration_value) + return config_items + + @staticmethod + def get_configuration_overrides(context, configuration_id): + """Gets the overrides dict to apply to an instance""" + overrides = {} + if configuration_id: + config_items = Configuration.load_items(context, + id=configuration_id) + + for i in config_items: + overrides[i.configuration_key] = i.configuration_value + return overrides + + @staticmethod + def save(context, configuration, configuration_items, instances): + DBConfiguration.save(configuration) + for item in configuration_items: + item["deleted_at"] = None + ConfigurationParameter.save(item) + + items = Configuration.load_items(context, configuration.id) + + for instance in instances: + LOG.debug(_("applying to instance: %s") % instance.id) + overrides = {} + for i in items: + overrides[i.configuration_key] = i.configuration_value + + task_api.API(context).update_overrides(instance.id, overrides) + + +class DBConfiguration(dbmodels.DatabaseModelBase): + _data_fields = ['name', 'description', 'tenant_id', 'datastore_version_id', + 'deleted', 'deleted_at'] + + +class ConfigurationParameter(dbmodels.DatabaseModelBase): + _data_fields = ['configuration_id', 'configuration_key', + 'configuration_value', 'deleted', + 'deleted_at'] + + def __hash__(self): + return self.configuration_key.__hash__() + + +def persisted_models(): + return { + 'configurations': DBConfiguration, + 'configuration_parameters': ConfigurationParameter + } diff --git a/trove/configuration/service.py b/trove/configuration/service.py new file mode 100644 index 0000000000..891b07307f --- /dev/null +++ b/trove/configuration/service.py @@ -0,0 +1,287 @@ +# Copyright 2014 Rackspace +# 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. + +from datetime import datetime +from trove.common import cfg +from trove.common import configurations +from trove.common import exception +from trove.common import pagination +from trove.common import wsgi +from trove.configuration import models +from trove.configuration import views +from trove.configuration.models import ConfigurationParameter +from trove.datastore import models as ds_models +from trove.openstack.common import log as logging +from trove.openstack.common.gettextutils import _ +from trove.instance import models as instances_models +import trove.common.apischema as apischema + + +CONF = cfg.CONF +LOG = logging.getLogger(__name__) + + +class ConfigurationsController(wsgi.Controller): + + schemas = apischema.configuration + + def index(self, req, tenant_id): + context = req.environ[wsgi.CONTEXT_KEY] + configs, marker = models.Configurations.load(context) + view = views.ConfigurationsView(configs) + paged = pagination.SimplePaginatedDataView(req.url, 'configurations', + view, marker) + return wsgi.Result(paged.data(), 200) + + def show(self, req, tenant_id, id): + context = req.environ[wsgi.CONTEXT_KEY] + configuration = models.Configuration.load(context, id) + configuration_items = models.Configuration.load_items(context, id) + + return wsgi.Result(views.DetailedConfigurationView( + configuration, + configuration_items).data(), 200) + + def instances(self, req, tenant_id, id): + context = req.environ[wsgi.CONTEXT_KEY] + configuration = models.Configuration.load(context, id) + instances = instances_models.DBInstance.find_all( + tenant_id=context.tenant, + configuration_id=configuration.id, + deleted=False) + limit = int(context.limit or CONF.instances_page_size) + if limit > CONF.instances_page_size: + limit = CONF.instances_page_size + data_view = instances_models.DBInstance.find_by_pagination( + 'instances', instances, "foo", + limit=limit, + marker=context.marker) + view = views.DetailedConfigurationInstancesView(data_view.collection) + paged = pagination.SimplePaginatedDataView(req.url, 'instances', view, + data_view.next_page_marker) + return wsgi.Result(paged.data(), 200) + + def create(self, req, body, tenant_id): + LOG.debug(_("req : '%s'\n\n") % req) + LOG.debug(_("body : '%s'\n\n") % req) + + name = body['configuration']['name'] + description = body['configuration'].get('description') + values = body['configuration']['values'] + + datastore_args = body['configuration'].get('datastore', {}) + datastore, datastore_version = ( + ds_models.get_datastore_version(**datastore_args)) + + configItems = [] + if values: + # validate that the values passed in are permitted by the operator. + ConfigurationsController._validate_configuration( + body['configuration']['values'], + datastore_manager=datastore_version.manager) + + for k, v in values.iteritems(): + configItems.append(ConfigurationParameter( + configuration_key=k, + configuration_value=v)) + + cfg_group = models.Configuration.create(name, description, tenant_id, + datastore.id, + datastore_version.id) + cfg_group_items = models.Configuration.create_items(cfg_group.id, + values) + view_data = views.DetailedConfigurationView(cfg_group, + cfg_group_items) + return wsgi.Result(view_data.data(), 200) + + def delete(self, req, tenant_id, id): + context = req.environ[wsgi.CONTEXT_KEY] + group = models.Configuration.load(context, id) + instances = instances_models.DBInstance.find_all( + tenant_id=context.tenant, + configuration_id=id, + deleted=False).all() + if instances: + raise exception.InstanceAssignedToConfiguration() + models.Configuration.delete(context, group) + return wsgi.Result(None, 202) + + def update(self, req, body, tenant_id, id): + LOG.info(_("Updating configuration for tenant id %s") % tenant_id) + context = req.environ[wsgi.CONTEXT_KEY] + group = models.Configuration.load(context, id) + instances = instances_models.DBInstance.find_all( + tenant_id=context.tenant, + configuration_id=id, + deleted=False).all() + + # if name/description are provided in the request body, update the + # model with these values as well. + if 'name' in body['configuration']: + group.name = body['configuration']['name'] + + if 'description' in body['configuration']: + group.description = body['configuration']['description'] + + items = self._configuration_items_list(group, body['configuration']) + deleted_at = datetime.utcnow() + models.Configuration.remove_all_items(context, group.id, deleted_at) + LOG.info(_("loaded configuration instances: %s") % instances) + models.Configuration.save(context, group, items, instances) + return wsgi.Result(None, 202) + + def edit(self, req, body, tenant_id, id): + context = req.environ[wsgi.CONTEXT_KEY] + group = models.Configuration.load(context, id) + instances = instances_models.DBInstance.find_all( + tenant_id=context.tenant, + configuration_id=id, + deleted=False).all() + LOG.info(_("loaded configuration instances: %s") % instances) + items = self._configuration_items_list(group, body['configuration']) + models.Configuration.save(context, group, items, instances) + + def _configuration_items_list(self, group, configuration): + ds_version_id = group.datastore_version_id + ds_version = ds_models.DatastoreVersion.load_by_uuid(ds_version_id) + items = [] + LOG.info(_("loaded configuration group: %s") % group) + if 'values' in configuration: + # validate that the values passed in are permitted by the operator. + ConfigurationsController._validate_configuration( + configuration['values'], datastore_manager=ds_version.manager) + for k, v in configuration['values'].iteritems(): + items.append(ConfigurationParameter(configuration_id=group.id, + configuration_key=k, + configuration_value=v, + deleted=False)) + return items + + @staticmethod + def _validate_configuration(values, datastore_manager=None): + rules = configurations.get_validation_rules( + datastore_manager=datastore_manager) + + LOG.info(_("Validating configuration values")) + for k, v in values.iteritems(): + # get the validation rule dictionary, which will ensure there is a + # rule for the given key name. An exception will be thrown if no + # valid rule is located. + rule = ConfigurationsController._get_item( + k, rules['configuration-parameters']) + + if rule.get('deleted_at'): + raise exception.ConfigurationParameterDeleted( + parameter_name=rule.get('name'), + parameter_deleted_at=rule.get('deleted_at')) + + # type checking + valueType = rule.get('type') + + if not isinstance(v, ConfigurationsController._find_type( + valueType)): + output = {"key": k, "type": valueType} + msg = _("The value provided for the configuration " + "parameter %(key)s is not of type %(type)s.") % output + raise exception.UnprocessableEntity(message=msg) + + # integer min/max checking + if isinstance(v, int): + try: + min_value = int(rule.get('min')) + except ValueError: + raise exception.TroveError(_( + "Invalid or unsupported min value defined in the " + "configuration-parameters configuration file. " + "Expected integer.")) + if v < min_value: + output = {"key": k, "min": min_value} + message = _("The value for the configuration parameter " + "%(key)s is less than the minimum allowed: " + "%(min)s") % output + raise exception.UnprocessableEntity(message=message) + + try: + max_value = int(rule.get('max')) + except ValueError: + raise exception.TroveError(_( + "Invalid or unsupported max value defined in the " + "configuration-parameters configuration file. " + "Expected integer.")) + if v > max_value: + output = {"key": k, "max": max_value} + message = _("The value for the configuration parameter " + "%(key)s is greater than the maximum " + "allowed: %(max)s") % output + raise exception.UnprocessableEntity(message=message) + + @staticmethod + def _find_type(valueType): + if valueType == "boolean": + return bool + elif valueType == "string": + return basestring + elif valueType == "integer": + return int + else: + raise exception.TroveError(_( + "Invalid or unsupported type defined in the " + "configuration-parameters configuration file.")) + + @staticmethod + def _get_item(key, dictList): + for item in dictList: + if key == item.get('name'): + return item + raise exception.UnprocessableEntity( + message=_("%s is not a supported configuration parameter.") % key) + + +class ParametersController(wsgi.Controller): + def index(self, req, tenant_id, datastore, id): + ds, ds_version = ds_models.get_datastore_version( + type=datastore, version=id) + rules = configurations.get_validation_rules( + datastore_manager=ds_version.manager) + return wsgi.Result(views.ConfigurationParametersView(rules).data(), + 200) + + def show(self, req, tenant_id, datastore, id, name): + ds, ds_version = ds_models.get_datastore_version( + type=datastore, version=id) + rules = configurations.get_validation_rules( + datastore_manager=ds_version.manager) + for rule in rules['configuration-parameters']: + if rule['name'] == name: + return wsgi.Result( + views.ConfigurationParametersView(rule).data(), 200) + raise exception.ConfigKeyNotFound(key=name) + + def index_by_version(self, req, tenant_id, version): + ds_version = ds_models.DatastoreVersion.load_by_uuid(version) + rules = configurations.get_validation_rules( + datastore_manager=ds_version.manager) + return wsgi.Result(views.ConfigurationParametersView(rules).data(), + 200) + + def show_by_version(self, req, tenant_id, version, name): + ds_version = ds_models.DatastoreVersion.load_by_uuid(version) + rules = configurations.get_validation_rules( + datastore_manager=ds_version.manager) + for rule in rules['configuration-parameters']: + if rule['name'] == name: + return wsgi.Result( + views.ConfigurationParametersView(rule).data(), 200) + raise exception.ConfigKeyNotFound(key=name) diff --git a/trove/configuration/views.py b/trove/configuration/views.py new file mode 100644 index 0000000000..d410b51ea1 --- /dev/null +++ b/trove/configuration/views.py @@ -0,0 +1,107 @@ +# Copyright 2014 Rackspace +# 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. + +from trove.openstack.common import log as logging + +LOG = logging.getLogger(__name__) + + +class ConfigurationView(object): + + def __init__(self, configuration): + self.configuration = configuration + + def data(self): + configuration_dict = { + "id": self.configuration.id, + "name": self.configuration.name, + "description": self.configuration.description, + "datastore_version_id": self.configuration.datastore_version_id, + } + + return {"configuration": configuration_dict} + + +class ConfigurationsView(object): + + def __init__(self, configurations): + self.configurations = configurations + + def data(self): + data = [] + + for configuration in self.configurations: + data.append(self.data_for_configuration(configuration)) + + return {"configurations": data} + + def data_for_configuration(self, configuration): + view = ConfigurationView(configuration) + return view.data()['configuration'] + + +class DetailedConfigurationInstancesView(object): + + def __init__(self, instances): + self.instances = instances + + def instance_data(self): + instances_list = [] + if self.instances: + for instance in self.instances: + instances_list.append( + { + "id": instance.id, + "name": instance.name + } + ) + return instances_list + + def data(self): + + return {"instances": self.instance_data()} + + +class DetailedConfigurationView(object): + + def __init__(self, configuration, configuration_items): + self.configuration = configuration + self.configuration_items = configuration_items + + def data(self): + values = {} + + for configItem in self.configuration_items: + key = configItem.configuration_key + value = configItem.configuration_value + values[key] = value + configuration_dict = { + "id": self.configuration.id, + "name": self.configuration.name, + "description": self.configuration.description, + "values": values, + "datastore_version_id": self.configuration.datastore_version_id, + } + + return {"configuration": configuration_dict} + + +class ConfigurationParametersView(object): + + def __init__(self, configuration_parameters): + self.configuration_parameters = configuration_parameters + + def data(self): + return self.configuration_parameters diff --git a/trove/db/sqlalchemy/mappers.py b/trove/db/sqlalchemy/mappers.py index 835eec8d57..97d12ed387 100644 --- a/trove/db/sqlalchemy/mappers.py +++ b/trove/db/sqlalchemy/mappers.py @@ -55,6 +55,10 @@ def map(engine, models): orm.mapper(models['security_group_instance_association'], Table('security_group_instance_associations', meta, autoload=True)) + orm.mapper(models['configurations'], + Table('configurations', meta, autoload=True)) + orm.mapper(models['configuration_parameters'], + Table('configuration_parameters', meta, autoload=True)) def mapping_exists(model): diff --git a/trove/db/sqlalchemy/migrate_repo/versions/020_configurations.py b/trove/db/sqlalchemy/migrate_repo/versions/020_configurations.py new file mode 100644 index 0000000000..a6b654d131 --- /dev/null +++ b/trove/db/sqlalchemy/migrate_repo/versions/020_configurations.py @@ -0,0 +1,65 @@ +# Copyright 2014 Rackspace +# 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. + +from sqlalchemy import ForeignKey +from sqlalchemy.schema import Column +from sqlalchemy.schema import MetaData + +from trove.db.sqlalchemy.migrate_repo.schema import create_tables +from trove.db.sqlalchemy.migrate_repo.schema import DateTime +from trove.db.sqlalchemy.migrate_repo.schema import Boolean +from trove.db.sqlalchemy.migrate_repo.schema import String +from trove.db.sqlalchemy.migrate_repo.schema import Table + +meta = MetaData() + +configurations = Table( + 'configurations', + meta, + Column('id', String(36), primary_key=True, nullable=False), + Column('name', String(64), nullable=False), + Column('description', String(256)), + Column('tenant_id', String(36), nullable=False), + Column('datastore_version_id', String(36), nullable=False), + Column('deleted', Boolean(), nullable=False, default=False), + Column('deleted_at', DateTime()), +) + +configuration_parameters = Table( + 'configuration_parameters', + meta, + Column('configuration_id', String(36), ForeignKey("configurations.id"), + nullable=False, primary_key=True), + Column('configuration_key', String(128), nullable=False, primary_key=True), + Column('configuration_value', String(128)), + Column('deleted', Boolean(), nullable=False, default=False), + Column('deleted_at', DateTime()), +) + + +def upgrade(migrate_engine): + meta.bind = migrate_engine + create_tables([configurations]) + create_tables([configuration_parameters]) + + instances = Table('instances', meta, autoload=True) + instances.create_column(Column('configuration_id', String(36), + ForeignKey("configurations.id"))) + + +def downgrade(migrate_engine): + meta.bind = migrate_engine + # Not dropping the tables for concern if rollback needed would cause + # consumers to recreate configurations. diff --git a/trove/db/sqlalchemy/session.py b/trove/db/sqlalchemy/session.py index 4cd94b07ac..93e0e832b6 100644 --- a/trove/db/sqlalchemy/session.py +++ b/trove/db/sqlalchemy/session.py @@ -49,6 +49,7 @@ def configure_db(options, models_mapper=None): from trove.quota import models as quota_models from trove.backup import models as backup_models from trove.extensions.security_group import models as secgrp_models + from trove.configuration import models as configurations_models model_modules = [ base_models, @@ -59,6 +60,7 @@ def configure_db(options, models_mapper=None): quota_models, backup_models, secgrp_models, + configurations_models, ] models = {} diff --git a/trove/guestagent/api.py b/trove/guestagent/api.py index 3cb88c72db..285e98e2b1 100644 --- a/trove/guestagent/api.py +++ b/trove/guestagent/api.py @@ -220,7 +220,8 @@ class API(proxy.RpcProxy): def prepare(self, memory_mb, packages, databases, users, device_path='/dev/vdb', mount_point='/mnt/volume', - backup_info=None, config_contents=None, root_password=None): + backup_info=None, config_contents=None, root_password=None, + overrides=None): """Make an asynchronous call to prepare the guest as a database container optionally includes a backup id for restores """ @@ -229,7 +230,8 @@ class API(proxy.RpcProxy): "prepare", packages=packages, databases=databases, memory_mb=memory_mb, users=users, device_path=device_path, mount_point=mount_point, backup_info=backup_info, - config_contents=config_contents, root_password=root_password) + config_contents=config_contents, root_password=root_password, + overrides=overrides) def restart(self): """Restart the MySQL server.""" @@ -301,3 +303,13 @@ class API(proxy.RpcProxy): 'device': device_path, 'id': self.id}) self._call("resize_fs", AGENT_LOW_TIMEOUT, device_path=device_path, mount_point=mount_point) + + def update_overrides(self, overrides, remove=False): + LOG.debug(_("Updating overrides on Instance %s"), self.id) + LOG.debug(_("Updating overrides values %s") % overrides) + self._cast("update_overrides", overrides=overrides, remove=remove) + + def apply_overrides(self, overrides): + LOG.debug(_("Applying overrides on Instance %s"), self.id) + LOG.debug(_("Applying overrides values %s") % overrides) + self._cast("apply_overrides", overrides=overrides) diff --git a/trove/guestagent/common/sql_query.py b/trove/guestagent/common/sql_query.py index 19e7c2f703..ad2d9489a2 100644 --- a/trove/guestagent/common/sql_query.py +++ b/trove/guestagent/common/sql_query.py @@ -425,6 +425,25 @@ class DropUser(object): return "DROP USER `%s`@`%s`;" % (self.user, self.host) +class SetServerVariable(object): + + def __init__(self, key, value): + self.key = key + self.value = value + + def __repr__(self): + return str(self) + + def __str__(self): + if self.value is True: + return "SET GLOBAL %s=%s" % (self.key, 1) + elif self.value is False: + return "SET GLOBAL %s=%s" % (self.key, 0) + elif self.value is None: + return "SET GLOBAL %s" % (self.key) + else: + return "SET GLOBAL %s=%s" % (self.key, self.value) + ### Miscellaneous queries that need no parameters. FLUSH = "FLUSH PRIVILEGES;" diff --git a/trove/guestagent/datastore/mysql/manager.py b/trove/guestagent/datastore/mysql/manager.py index d1425b3d83..de907144a2 100644 --- a/trove/guestagent/datastore/mysql/manager.py +++ b/trove/guestagent/datastore/mysql/manager.py @@ -105,7 +105,7 @@ class Manager(periodic_task.PeriodicTasks): def prepare(self, context, packages, databases, memory_mb, users, device_path=None, mount_point=None, backup_info=None, - config_contents=None, root_password=None): + config_contents=None, root_password=None, overrides=None): """Makes ready DBAAS on a Guest container.""" MySqlAppStatus.get().begin_install() # status end_mysql_install set with secure() @@ -127,7 +127,7 @@ class Manager(periodic_task.PeriodicTasks): self._perform_restore(backup_info, context, CONF.mount_point, app) LOG.info(_("Securing mysql now.")) - app.secure(config_contents) + app.secure(config_contents, overrides) enable_root_on_restore = (backup_info and MySqlAdmin().is_root_enabled()) if root_password and not backup_info: @@ -190,3 +190,11 @@ class Manager(periodic_task.PeriodicTasks): device = volume.VolumeDevice(device_path) device.resize_fs(mount_point) LOG.debug(_("Resized the filesystem")) + + def update_overrides(self, context, overrides, remove=False): + app = MySqlApp(MySqlAppStatus.get()) + app.update_overrides(overrides, remove=remove) + + def apply_overrides(self, context, overrides): + app = MySqlApp(MySqlAppStatus.get()) + app.apply_overrides(overrides) diff --git a/trove/guestagent/datastore/mysql/service.py b/trove/guestagent/datastore/mysql/service.py index d187cc7e92..b21b56f799 100644 --- a/trove/guestagent/datastore/mysql/service.py +++ b/trove/guestagent/datastore/mysql/service.py @@ -48,6 +48,7 @@ TMP_MYCNF = "/tmp/my.cnf.tmp" MYSQL_BASE_DIR = "/var/lib/mysql" CONF = cfg.CONF + INCLUDE_MARKER_OPERATORS = { True: ">=", False: ">" @@ -56,6 +57,8 @@ INCLUDE_MARKER_OPERATORS = { MYSQL_CONFIG = "/etc/mysql/my.cnf" MYSQL_SERVICE_CANDIDATES = ["mysql", "mysqld", "mysql-server"] MYSQL_BIN_CANDIDATES = ["/usr/sbin/mysqld", "/usr/libexec/mysqld"] +MYCNF_OVERRIDES = "/etc/mysql/conf.d/overrides.cnf" +MYCNF_OVERRIDES_TMP = "/tmp/overrides.cnf.tmp" # Create a package impl @@ -348,7 +351,7 @@ class MySqlAdmin(object): user = models.MySQLUser() try: user.name = username # Could possibly throw a BadRequest here. - except exceptions.ValueError as ve: + except exception.ValueError as ve: raise exception.BadRequest(_("Username %(user)s is not valid" ": %(reason)s") % {'user': username, 'reason': ve.message} @@ -597,7 +600,7 @@ class MySqlApp(object): def complete_install_or_restart(self): self.status.end_install_or_restart() - def secure(self, config_contents): + def secure(self, config_contents, overrides): LOG.info(_("Generating admin password...")) admin_password = utils.generate_random_password() clear_expired_password() @@ -608,7 +611,7 @@ class MySqlApp(object): self._create_admin_user(client, admin_password) self.stop_db() - self._write_mycnf(admin_password, config_contents) + self._write_mycnf(admin_password, config_contents, overrides) self.start_mysql() LOG.info(_("Dbaas secure complete.")) @@ -691,10 +694,41 @@ class MySqlApp(object): finally: self.status.end_install_or_restart() + def update_overrides(self, overrides_file, remove=False): + """ + This function will either update or remove the MySQL overrides.cnf file + If remove is set to True the function will remove the overrides file. + + :param overrides: + :param remove: + :return: + """ + + if overrides_file: + LOG.debug(_("writing new overrides.cnf config file")) + self._write_config_overrides(overrides_file) + if remove: + LOG.debug(_("removing overrides.cnf config file")) + self._remove_overrides() + + def apply_overrides(self, overrides): + LOG.debug(_("applying overrides to mysql")) + with LocalSqlClient(get_engine()) as client: + LOG.debug(_("updating overrides values in running daemon")) + for k, v in overrides.iteritems(): + q = sql_query.SetServerVariable(key=k, value=v) + t = text(str(q)) + try: + client.execute(t) + except exc.OperationalError: + output = {'key': k, 'value': v} + LOG.exception(_("Unable to set %(key)s with value " + "%(value)s") % output) + def _replace_mycnf_with_template(self, template_path, original_path): - LOG.debug("replacing the mycnf with template") - LOG.debug("template_path(%s) original_path(%s)" - % (template_path, original_path)) + LOG.debug(_("replacing the mycnf with template")) + LOG.debug(_("template_path(%(template)s) original_path(%(origin)s)") + % {"template": template_path, "origin": original_path}) if os.path.isfile(template_path): if os.path.isfile(original_path): utils.execute_with_timeout( @@ -739,7 +773,7 @@ class MySqlApp(object): if "No such file or directory" not in str(pe): raise - def _write_mycnf(self, admin_password, config_contents): + def _write_mycnf(self, admin_password, config_contents, overrides=None): """ Install the set of mysql my.cnf templates. Update the os_admin user and password to the my.cnf @@ -762,6 +796,28 @@ class MySqlApp(object): self.wipe_ib_logfiles() + # write configuration file overrides + if overrides: + self._write_config_overrides(overrides) + + def _write_config_overrides(self, overrideValues): + LOG.info(_("Writing new temp overrides.cnf file.")) + + with open(MYCNF_OVERRIDES_TMP, 'w') as overrides: + overrides.write(overrideValues) + LOG.info(_("Moving overrides.cnf into correct location.")) + utils.execute_with_timeout("sudo", "mv", MYCNF_OVERRIDES_TMP, + MYCNF_OVERRIDES) + + LOG.info(_("Setting permissions on overrides.cnf")) + utils.execute_with_timeout("sudo", "chmod", "0711", + MYCNF_OVERRIDES) + + def _remove_overrides(self): + LOG.info(_("Removing overrides configuration file")) + if os.path.exists(MYCNF_OVERRIDES): + utils.execute_with_timeout("sudo", "rm", MYCNF_OVERRIDES) + def start_mysql(self, update_db=False): LOG.info(_("Starting mysql...")) # This is the site of all the trouble in the restart tests. diff --git a/trove/instance/models.py b/trove/instance/models.py index f92f540b8b..92a19e7528 100644 --- a/trove/instance/models.py +++ b/trove/instance/models.py @@ -1,7 +1,8 @@ # vim: tabstop=4 shiftwidth=4 softtabstop=4 -# Copyright 2010-2011 OpenStack Foundation -# All Rights Reserved. +# Copyright 2010-2011 OpenStack Foundation +# Copyright 2013-2014 Rackspace Hosting +# 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 @@ -22,12 +23,15 @@ from datetime import datetime from novaclient import exceptions as nova_exceptions from trove.common import cfg from trove.common import exception -import trove.common.instance as rd_instance +from trove.common import template +from trove.common.configurations import do_configs_require_restart +import trove.common.instance as tr_instance from trove.common.remote import create_dns_client from trove.common.remote import create_guest_client from trove.common.remote import create_nova_client from trove.common.remote import create_cinder_client from trove.common import utils +from trove.configuration.models import Configuration from trove.extensions.security_group.models import SecurityGroup from trove.db import get_db_api from trove.db import models as dbmodels @@ -74,6 +78,7 @@ class InstanceStatus(object): BACKUP = "BACKUP" SHUTDOWN = "SHUTDOWN" ERROR = "ERROR" + RESTART_REQUIRED = "RESTART_REQUIRED" def validate_volume_size(size): @@ -226,6 +231,8 @@ class SimpleInstance(object): return InstanceStatus.REBOOT if 'RESIZING' == ACTION: return InstanceStatus.RESIZE + if 'RESTART_REQUIRED' == ACTION: + return InstanceStatus.RESTART_REQUIRED ### Check for server status. if self.db_info.server_status in ["BUILD", "ERROR", "REBOOT", @@ -254,10 +261,10 @@ class SimpleInstance(object): ### Check against the service status. # The service is only paused during a reboot. - if rd_instance.ServiceStatuses.PAUSED == self.service_status.status: + if tr_instance.ServiceStatuses.PAUSED == self.service_status.status: return InstanceStatus.REBOOT # If the service status is NEW, then we are building. - if rd_instance.ServiceStatuses.NEW == self.service_status.status: + if tr_instance.ServiceStatuses.NEW == self.service_status.status: return InstanceStatus.BUILD # For everything else we can look at the service status mapping. @@ -287,6 +294,12 @@ class SimpleInstance(object): def root_password(self): return self.root_pass + @property + def configuration(self): + if self.db_info.configuration_id is not None: + return Configuration.load(self.context, + self.db_info.configuration_id) + class DetailInstance(SimpleInstance): """A detailed view of an Instnace. @@ -407,7 +420,8 @@ class BaseInstance(SimpleInstance): LOG.debug(_(" ... deleting compute id = %s") % self.db_info.compute_instance_id) LOG.debug(_(" ... setting status to DELETING.")) - self.update_db(task_status=InstanceTasks.DELETING) + self.update_db(task_status=InstanceTasks.DELETING, + configuration_id=None) task_api.API(self.context).delete_instance(self.id) deltas = {'instances': -1} @@ -458,7 +472,7 @@ class BaseInstance(SimpleInstance): def set_servicestatus_deleted(self): del_instance = InstanceServiceStatus.find_by(instance_id=self.id) - del_instance.set_status(rd_instance.ServiceStatuses.DELETED) + del_instance.set_status(tr_instance.ServiceStatuses.DELETED) del_instance.save() @property @@ -491,7 +505,7 @@ class Instance(BuiltInstance): @classmethod def create(cls, context, name, flavor_id, image_id, databases, users, datastore, datastore_version, volume_size, backup_id, - availability_zone=None, nics=None): + availability_zone=None, nics=None, configuration_id=None): client = create_nova_client(context) try: @@ -533,14 +547,21 @@ class Instance(BuiltInstance): volume_size=volume_size, datastore_version_id= datastore_version.id, - task_status=InstanceTasks.BUILDING) + task_status=InstanceTasks.BUILDING, + configuration_id=configuration_id) LOG.debug(_("Tenant %(tenant)s created new " "Trove instance %(db)s...") % {'tenant': context.tenant, 'db': db_info.id}) + # if a configuration group is associated with an instance, + # generate an overrides dict to pass into the instance creation + # method + + overrides = Configuration.get_configuration_overrides( + context, configuration_id) service_status = InstanceServiceStatus.create( instance_id=db_info.id, - status=rd_instance.ServiceStatuses.NEW) + status=tr_instance.ServiceStatuses.NEW) if CONF.trove_dns_support: dns_client = create_dns_client(context) @@ -558,7 +579,9 @@ class Instance(BuiltInstance): datastore_version.packages, volume_size, backup_id, availability_zone, - root_password, nics) + root_password, + nics, + overrides) return SimpleInstance(context, db_info, service_status, root_password) @@ -567,6 +590,17 @@ class Instance(BuiltInstance): deltas, _create_resources) + def get_flavor(self): + client = create_nova_client(self.context) + return client.flavors.get(self.flavor_id) + + def get_default_configration_template(self): + flavor = self.get_flavor() + LOG.debug("flavor: %s" % flavor) + config = template.SingleInstanceConfigTemplate( + self.ds_version.manager, flavor, id) + return config.render_dict() + def resize_flavor(self, new_flavor_id): self.validate_can_perform_action() LOG.debug("resizing instance %s flavor to %s" @@ -654,21 +688,73 @@ class Instance(BuiltInstance): """ Raises exception if an instance action cannot currently be performed. """ + # cases where action cannot be performed if self.db_info.server_status != 'ACTIVE': status = self.db_info.server_status - elif self.db_info.task_status != InstanceTasks.NONE: + elif (self.db_info.task_status != InstanceTasks.NONE and + self.db_info.task_status != InstanceTasks.RESTART_REQUIRED): status = self.db_info.task_status elif not self.service_status.status.action_is_allowed: status = self.status elif Backup.running(self.id): status = InstanceStatus.BACKUP else: + # action can be performed return + msg = ("Instance is not currently available for an action to be " "performed (status was %s)." % status) LOG.error(msg) raise exception.UnprocessableEntity(msg) + def unassign_configuration(self): + LOG.debug(_("Unassigning the configuration from the instance %s") + % self.id) + LOG.debug(_("Unassigning the configuration id %s") + % self.configuration.id) + if self.configuration and self.configuration.id: + flavor = self.get_flavor() + config_id = self.configuration.id + task_api.API(self.context).unassign_configuration(self.id, + flavor, + config_id) + else: + LOG.debug("no configuration found on instance skipping.") + + def assign_configuration(self, configuration_id): + try: + configuration = Configuration.load(self.context, configuration_id) + except exception.ModelNotFoundError: + raise exception.NotFound( + message='Configuration group id: %s could not be found' + % configuration_id) + + config_ds_v = configuration.datastore_version_id + inst_ds_v = self.db_info.datastore_version_id + if (config_ds_v != inst_ds_v): + raise exception.ConfigurationDatastoreNotMatchInstance( + config_datastore_version=config_ds_v, + instance_datastore_version=inst_ds_v) + + overrides = Configuration.get_configuration_overrides( + self.context, configuration.id) + + LOG.info(overrides) + + self.update_overrides(overrides) + self.update_db(configuration_id=configuration.id) + + def update_overrides(self, overrides): + LOG.debug(_("Updating or removing overrides for instance %s") + % self.id) + need_restart = do_configs_require_restart( + overrides, datastore_manager=self.ds_version.manager) + LOG.debug(_("config overrides has non-dynamic settings, " + "requires a restart: %s") % need_restart) + if need_restart: + self.update_db(task_status=InstanceTasks.RESTART_REQUIRED) + task_api.API(self.context).update_overrides(self.id, overrides) + def create_server_list_matcher(server_list): # Returns a method which finds a server from the given list. @@ -765,7 +851,7 @@ class DBInstance(dbmodels.DatabaseModelBase): _data_fields = ['name', 'created', 'compute_instance_id', 'task_id', 'task_description', 'task_start_time', 'volume_id', 'deleted', 'tenant_id', - 'datastore_version_id'] + 'datastore_version_id', 'configuration_id'] def __init__(self, task_status, **kwargs): kwargs["task_id"] = task_status.code @@ -803,11 +889,11 @@ class InstanceServiceStatus(dbmodels.DatabaseModelBase): def _validate(self, errors): if self.status is None: errors['status'] = "Cannot be none." - if rd_instance.ServiceStatus.from_code(self.status_id) is None: + if tr_instance.ServiceStatus.from_code(self.status_id) is None: errors['status_id'] = "Not valid." def get_status(self): - return rd_instance.ServiceStatus.from_code(self.status_id) + return tr_instance.ServiceStatus.from_code(self.status_id) def set_status(self, value): self.status_id = value.code @@ -827,4 +913,4 @@ def persisted_models(): } -MYSQL_RESPONSIVE_STATUSES = [rd_instance.ServiceStatuses.RUNNING] +MYSQL_RESPONSIVE_STATUSES = [tr_instance.ServiceStatuses.RUNNING] diff --git a/trove/instance/service.py b/trove/instance/service.py index b69d8c774e..5d0575e461 100644 --- a/trove/instance/service.py +++ b/trove/instance/service.py @@ -38,6 +38,7 @@ LOG = logging.getLogger(__name__) class InstanceController(wsgi.Controller): + """Controller for instance functionality""" schemas = apischema.instance.copy() if not CONF.trove_volume_support: @@ -189,6 +190,8 @@ class InstanceController(wsgi.Controller): name = body['instance']['name'] flavor_ref = body['instance']['flavorRef'] flavor_id = utils.get_id_from_href(flavor_ref) + + configuration = self._configuration_parse(context, body) databases = populate_validated_databases( body['instance'].get('databases', [])) database_names = [database.get('_name', '') for database in databases] @@ -224,7 +227,50 @@ class InstanceController(wsgi.Controller): image_id, databases, users, datastore, datastore_version, volume_size, backup_id, - availability_zone, nics) + availability_zone, nics, + configuration) view = views.InstanceDetailView(instance, req=req) return wsgi.Result(view.data(), 200) + + def _configuration_parse(self, context, body): + if 'configuration' in body['instance']: + configuration_ref = body['instance']['configuration'] + if configuration_ref: + configuration_id = utils.get_id_from_href(configuration_ref) + return configuration_id + + def update(self, req, id, body, tenant_id): + """Updates the instance to attach/detach configuration.""" + LOG.info(_("Updating instance for tenant id %s") % tenant_id) + LOG.info(_("req: %s") % req) + LOG.info(_("body: %s") % body) + context = req.environ[wsgi.CONTEXT_KEY] + + instance = models.Instance.load(context, id) + + # if configuration is set, then we will update the instance to use + # the new configuration. If configuration is empty, we want to + # disassociate the instance from the configuration group and remove the + # active overrides file. + + configuration_id = self._configuration_parse(context, body) + + if configuration_id: + instance.assign_configuration(configuration_id) + else: + instance.unassign_configuration() + return wsgi.Result(None, 202) + + def configuration(self, req, tenant_id, id): + """ + Returns the default configuration template applied to the instance. + """ + LOG.debug("getting default configuration for the instance(%s)" % id) + context = req.environ[wsgi.CONTEXT_KEY] + instance = models.Instance.load(context, id) + LOG.debug("server: %s" % instance) + config = instance.get_default_configration_template() + LOG.debug("default config for instance is: %s" % config) + return wsgi.Result(views.DefaultConfigurationView( + config).data(), 200) diff --git a/trove/instance/tasks.py b/trove/instance/tasks.py index a17116f9f4..704c418934 100644 --- a/trove/instance/tasks.py +++ b/trove/instance/tasks.py @@ -70,6 +70,8 @@ class InstanceTasks(object): RESIZING = InstanceTask(0x04, 'RESIZING', 'Resizing the instance.') BUILDING = InstanceTask(0x05, 'BUILDING', 'The instance is building.') MIGRATING = InstanceTask(0x06, 'MIGRATING', 'Migrating the instance.') + RESTART_REQUIRED = InstanceTask(0x07, 'RESTART_REQUIRED', + 'Instance requires a restart.') BUILDING_ERROR_DNS = InstanceTask(0x50, 'BUILDING', 'Build error: DNS.', is_error=True) diff --git a/trove/instance/views.py b/trove/instance/views.py index c8511eedf1..70cc068a44 100644 --- a/trove/instance/views.py +++ b/trove/instance/views.py @@ -75,6 +75,9 @@ class InstanceDetailView(InstanceView): result['instance']['datastore']['version'] = (self.instance. datastore_version.name) + if self.instance.configuration is not None: + result['instance']['configuration'] = (self. + _build_configuration_info()) if self.instance.hostname: result['instance']['hostname'] = self.instance.hostname else: @@ -96,6 +99,14 @@ class InstanceDetailView(InstanceView): return result + def _build_configuration_info(self): + return { + "id": self.instance.configuration.id, + "name": self.instance.configuration.name, + "links": create_links("configurations", self.req, + self.instance.configuration.id) + } + class InstancesView(object): """Shows a list of SimpleInstance objects.""" @@ -114,3 +125,14 @@ class InstancesView(object): def data_for_instance(self, instance): view = InstanceView(instance, req=self.req) return view.data()['instance'] + + +class DefaultConfigurationView(object): + def __init__(self, config): + self.config = config + + def data(self): + config_dict = {} + for key, val in self.config: + config_dict[key] = val + return {"instance": {"configuration": config_dict}} diff --git a/trove/taskmanager/api.py b/trove/taskmanager/api.py index 37d9234f0c..a03a199383 100644 --- a/trove/taskmanager/api.py +++ b/trove/taskmanager/api.py @@ -22,6 +22,7 @@ Routes all the requests to the task manager. from trove.common import cfg from trove.openstack.common.rpc import proxy from trove.openstack.common import log as logging +from trove.openstack.common.gettextutils import _ CONF = cfg.CONF @@ -102,7 +103,8 @@ class API(proxy.RpcProxy): def create_instance(self, instance_id, name, flavor, image_id, databases, users, datastore_manager, packages, volume_size, backup_id=None, - availability_zone=None, root_password=None, nics=None): + availability_zone=None, root_password=None, + nics=None, overrides=None): LOG.debug("Making async call to create instance %s " % instance_id) self.cast(self.context, self.make_msg("create_instance", @@ -117,4 +119,25 @@ class API(proxy.RpcProxy): volume_size=volume_size, backup_id=backup_id, availability_zone=availability_zone, - root_password=root_password, nics=nics)) + root_password=root_password, + nics=nics, + overrides=overrides)) + + def update_overrides(self, instance_id, overrides=None): + LOG.debug(_("Making async call to update configuration overrides for " + "instance %s") % instance_id) + + self.cast(self.context, + self.make_msg("update_overrides", + instance_id=instance_id, + overrides=overrides)) + + def unassign_configuration(self, instance_id, flavor, configuration_id): + LOG.debug(_("Making async call to unassign configuration for " + "instance %s") % instance_id) + + self.cast(self.context, + self.make_msg("unassign_configuration", + instance_id=instance_id, + flavor=self._transform_obj(flavor), + configuration_id=configuration_id)) diff --git a/trove/taskmanager/manager.py b/trove/taskmanager/manager.py index 448356b130..78cca4d9c9 100644 --- a/trove/taskmanager/manager.py +++ b/trove/taskmanager/manager.py @@ -83,12 +83,22 @@ class Manager(periodic_task.PeriodicTasks): def create_instance(self, context, instance_id, name, flavor, image_id, databases, users, datastore_manager, packages, volume_size, backup_id, availability_zone, - root_password, nics): + root_password, nics, overrides): instance_tasks = FreshInstanceTasks.load(context, instance_id) instance_tasks.create_instance(flavor, image_id, databases, users, datastore_manager, packages, volume_size, backup_id, - availability_zone, root_password, nics) + availability_zone, root_password, nics, + overrides) + + def update_overrides(self, context, instance_id, overrides): + instance_tasks = models.BuiltInstanceTasks.load(context, instance_id) + instance_tasks.update_overrides(overrides) + + def unassign_configuration(self, context, instance_id, flavor, + configuration_id): + instance_tasks = models.BuiltInstanceTasks.load(context, instance_id) + instance_tasks.unassign_configuration(flavor, configuration_id) if CONF.exists_notification_transformer: @periodic_task.periodic_task( diff --git a/trove/taskmanager/models.py b/trove/taskmanager/models.py index 6ec0199a5f..15db54286c 100644 --- a/trove/taskmanager/models.py +++ b/trove/taskmanager/models.py @@ -12,6 +12,7 @@ # License for the specific language governing permissions and limitations # under the License. +import re import traceback import os.path @@ -24,6 +25,7 @@ from trove.common import cfg from trove.common import template from trove.common import utils from trove.common.utils import try_recover +from trove.common.configurations import do_configs_require_restart from trove.common.exception import GuestError from trove.common.exception import GuestTimeout from trove.common.exception import PollTimeOut @@ -35,6 +37,7 @@ from trove.common.remote import create_dns_client from trove.common.remote import create_heat_client from trove.common.remote import create_cinder_client from trove.extensions.mysql import models as mysql_models +from trove.configuration.models import Configuration from trove.extensions.security_group.models import SecurityGroup from trove.extensions.security_group.models import SecurityGroupRule from swiftclient.client import ClientException @@ -143,11 +146,26 @@ class ConfigurationMixin(object): config.render() return config + def _render_override_config(self, datastore_manager, flavor, instance_id, + overrides=None): + config = template.OverrideConfigTemplate( + datastore_manager, flavor, instance_id) + config.render(overrides=overrides) + return config + + def _render_config_dict(self, datastore_manager, flavor, instance_id): + config = template.SingleInstanceConfigTemplate( + datastore_manager, flavor, instance_id) + ret = config.render_dict() + LOG.debug(_("the default template dict of mysqld section: %s") % ret) + return ret + class FreshInstanceTasks(FreshInstance, NotifyMixin, ConfigurationMixin): def create_instance(self, flavor, image_id, databases, users, datastore_manager, packages, volume_size, - backup_id, availability_zone, root_password, nics): + backup_id, availability_zone, root_password, nics, + overrides): LOG.debug(_("begin create_instance for id: %s") % self.id) security_groups = None @@ -197,6 +215,10 @@ class FreshInstanceTasks(FreshInstance, NotifyMixin, ConfigurationMixin): nics) config = self._render_config(datastore_manager, flavor, self.id) + config_overrides = self._render_override_config(datastore_manager, + None, + self.id, + overrides=overrides) backup_info = None if backup_id is not None: @@ -208,7 +230,8 @@ class FreshInstanceTasks(FreshInstance, NotifyMixin, ConfigurationMixin): } self._guest_prepare(flavor['ram'], volume_info, packages, databases, users, backup_info, - config.config_contents, root_password) + config.config_contents, root_password, + config_overrides.config_contents) if root_password: self.report_root_enabled() @@ -554,7 +577,8 @@ class FreshInstanceTasks(FreshInstance, NotifyMixin, ConfigurationMixin): def _guest_prepare(self, flavor_ram, volume_info, packages, databases, users, backup_info=None, - config_contents=None, root_password=None): + config_contents=None, root_password=None, + overrides=None): LOG.info(_("Entering guest_prepare")) # Now wait for the response from the create to do additional work self.guest.prepare(flavor_ram, packages, databases, users, @@ -562,7 +586,8 @@ class FreshInstanceTasks(FreshInstance, NotifyMixin, ConfigurationMixin): mount_point=volume_info['mount_point'], backup_info=backup_info, config_contents=config_contents, - root_password=root_password) + root_password=root_password, + overrides=overrides) def _create_dns_entry(self): LOG.debug(_("%(gt)s: Creating dns entry for instance: %(id)s") % @@ -756,6 +781,86 @@ class BuiltInstanceTasks(BuiltInstance, NotifyMixin, ConfigurationMixin): LOG.debug(_("Restarting FINALLY %s ") % self.id) self.update_db(task_status=inst_models.InstanceTasks.NONE) + def update_overrides(self, overrides, remove=False): + LOG.debug(_("Updating configuration overrides on instance %s") + % self.id) + LOG.debug(_("overrides: %s") % overrides) + LOG.debug(_("self.ds_version: %s") % self.ds_version.__dict__) + # todo(cp16net) How do we know what datastore type we have? + need_restart = do_configs_require_restart( + overrides, datastore_manager=self.ds_version.manager) + LOG.debug(_("do we need a restart?: %s") % need_restart) + if need_restart: + status = inst_models.InstanceTasks.RESTART_REQUIRED + self.update_db(task_status=status) + + config_overrides = self._render_override_config( + self.ds_version.manager, + None, + self.id, + overrides=overrides) + try: + self.guest.update_overrides(config_overrides.config_contents, + remove=remove) + self.guest.apply_overrides(overrides) + LOG.debug(_("Configuration overrides update successful.")) + except GuestError: + LOG.error(_("Failed to update configuration overrides.")) + + def unassign_configuration(self, flavor, configuration_id): + LOG.debug(_("Unassigning the configuration from the instance %s") + % self.id) + LOG.debug(_("Unassigning the configuration id %s") + % self.configuration.id) + + def _find_item(items, item_name): + LOG.debug(_("items: %s") % items) + LOG.debug(_("item_name: %s") % item_name) + # find the item in the list + for i in items: + if i[0] == item_name: + return i + + def _convert_value(value): + # split the value and the size e.g. 512M=['512','M'] + pattern = re.compile('(\d+)(\w+)') + split = pattern.findall(value) + if len(split) < 2: + return value + digits, size = split + conversions = { + 'K': 1024, + 'M': 1024 ** 2, + 'G': 1024 ** 3, + } + return str(int(digits) * conversions[size]) + + default_config = self._render_config_dict(self.ds_version.manager, + flavor, + self.id) + args = { + "ds_manager": self.ds_version.manager, + "config": default_config, + } + LOG.debug(_("default %(ds_manager)s section: %(config)s") % args) + LOG.debug(_("self.configuration: %s") % self.configuration.__dict__) + + overrides = {} + config_items = Configuration.load_items(self.context, configuration_id) + for item in config_items: + LOG.debug(_("finding item(%s)") % item.__dict__) + try: + key, val = _find_item(default_config, item.configuration_key) + except TypeError: + val = None + restart_required = inst_models.InstanceTasks.RESTART_REQUIRED + self.update_db(task_status=restart_required) + if val: + overrides[item.configuration_key] = _convert_value(val) + LOG.debug(_("setting the default variables in dict: %s") % overrides) + self.update_overrides(overrides, remove=True) + self.update_db(configuration_id=None) + def _refresh_compute_server_info(self): """Refreshes the compute server field.""" server = self.nova_client.servers.get(self.server.id) diff --git a/trove/templates/mysql/config.template b/trove/templates/mysql/config.template index a0276841d6..21d56ea2fa 100644 --- a/trove/templates/mysql/config.template +++ b/trove/templates/mysql/config.template @@ -12,7 +12,7 @@ datadir = /var/lib/mysql ####tmpdir = /tmp tmpdir = /var/tmp pid_file = /var/run/mysqld/mysqld.pid -skip-external-locking +skip-external-locking = 1 key_buffer_size = {{ 50 * flavor['ram']//512 }}M max_allowed_packet = {{ 1 * flavor['ram']//512 }}M thread_stack = 192K @@ -45,8 +45,8 @@ local-infile = 0 server_id = {{server_id}} [mysqldump] -quick -quote-names +quick = 1 +quote-names = 1 max_allowed_packet = 16M [isamchk] diff --git a/trove/templates/mysql/override.config.template b/trove/templates/mysql/override.config.template new file mode 100644 index 0000000000..c287e427d7 --- /dev/null +++ b/trove/templates/mysql/override.config.template @@ -0,0 +1,12 @@ +[mysqld] +{% for key, value in overrides.iteritems() -%} +{%- if value == True -%} +{{key}} = 1 +{%- elif value == False -%} +{{key}} = 0 +{%- elif value == "" -%} +{{key}} +{%- else -%} +{{key}}={{value}} +{%- endif %} +{% endfor %} diff --git a/trove/templates/mysql/validation-rules.json b/trove/templates/mysql/validation-rules.json new file mode 100644 index 0000000000..8b7153fcea --- /dev/null +++ b/trove/templates/mysql/validation-rules.json @@ -0,0 +1,224 @@ +{ + "configuration-parameters": [ + { + "name": "innodb_file_per_table", + "restart_required": true, + "max": 1, + "min": 0, + "type": "integer" + }, + { + "name": "autocommit", + "restart_required": false, + "max": 1, + "min": 0, + "type": "integer" + }, + { + "name": "local_infile", + "restart_required": false, + "max": 1, + "min": 0, + "type": "integer" + }, + { + "name": "key_buffer_size", + "restart_required": false, + "max": 4294967296, + "min": 0, + "type": "integer" + }, + { + "name": "connect_timeout", + "restart_required": false, + "max": 65535, + "min": 1, + "type": "integer" + }, + { + "name": "join_buffer_size", + "restart_required": false, + "max": 4294967296, + "min": 0, + "type": "integer" + }, + { + "name": "sort_buffer_size", + "restart_required": false, + "max": 18446744073709547520, + "min": 32768, + "type": "integer" + }, + { + "name": "innodb_buffer_pool_size", + "restart_required": true, + "max": 68719476736, + "min": 0, + "type": "integer" + }, + { + "name": "innodb_flush_log_at_trx_commit", + "restart_required": false, + "max": 2, + "min": 0, + "type": "integer" + }, + { + "name": "innodb_log_buffer_size", + "restart_required": true, + "max": 4294967296, + "min": 1048576, + "type": "integer" + }, + { + "name": "innodb_open_files", + "restart_required": true, + "max": 4294967296, + "min": 10, + "type": "integer" + }, + { + "name": "innodb_thread_concurrency", + "restart_required": false, + "max": 1000, + "min": 0, + "type": "integer" + }, + { + "name": "sync_binlog", + "restart_required": false, + "max": 18446744073709547520, + "min": 0, + "type": "integer" + }, + { + "name": "auto_increment_increment", + "restart_required": false, + "max": 65535, + "min": 1, + "type": "integer" + }, + { + "name": "auto_increment_offset", + "restart_required": false, + "max": 65535, + "min": 1, + "type": "integer" + }, + { + "name": "bulk_insert_buffer_size", + "restart_required": false, + "max": 18446744073709547520, + "min": 0, + "type": "integer" + }, + { + "name": "expire_logs_days", + "restart_required": false, + "max": 65535, + "min": 1, + "type": "integer" + }, + { + "name": "interactive_timeout", + "restart_required": false, + "max": 65535, + "min": 1, + "type": "integer" + }, + { + "name": "max_allowed_packet", + "restart_required": false, + "max": 1073741824, + "min": 1024, + "type": "integer" + }, + { + "name": "max_connect_errors", + "restart_required": false, + "max": 18446744073709547520, + "min": 1, + "type": "integer" + }, + { + "name": "max_connections", + "restart_required": false, + "max": 65535, + "min": 1, + "type": "integer" + }, + { + "name": "myisam_sort_buffer_size", + "restart_required": false, + "max": 18446744073709547520, + "min": 4, + "type": "integer" + }, + { + "name": "max_user_connections", + "restart_required": false, + "max": 100000, + "min": 1, + "type": "integer" + }, + { + "name": "server_id", + "restart_required": true, + "max": 100000, + "min": 1, + "type": "integer" + }, + { + "name": "wait_timeout", + "restart_required": false, + "max": 31536000, + "min": 1, + "type": "integer" + }, + { + "name": "character_set_client", + "restart_required": false, + "type": "string" + }, + { + "name": "character_set_connection", + "restart_required": false, + "type": "string" + }, + { + "name": "character_set_database", + "restart_required": false, + "type": "string" + }, + { + "name": "character_set_filesystem", + "restart_required": false, + "type": "string" + }, + { + "name": "character_set_results", + "restart_required": false, + "type": "string" + }, + { + "name": "character_set_server", + "restart_required": false, + "type": "string" + }, + { + "name": "collation_connection", + "restart_required": false, + "type": "string" + }, + { + "name": "collation_database", + "restart_required": false, + "type": "string" + }, + { + "name": "collation_server", + "restart_required": false, + "type": "string" + } + ] +} diff --git a/trove/tests/api/configurations.py b/trove/tests/api/configurations.py new file mode 100644 index 0000000000..46e1c2cd12 --- /dev/null +++ b/trove/tests/api/configurations.py @@ -0,0 +1,606 @@ +# Copyright 2014 Rackspace Hosting +# 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. + + +import json +from datetime import datetime +from proboscis import SkipTest +from proboscis import test +from proboscis.asserts import assert_equal +from proboscis.asserts import assert_raises +from proboscis.asserts import assert_true +from proboscis.asserts import assert_not_equal +from proboscis.decorators import time_out +from trove.common.utils import poll_until +from trove.tests.api.instances import assert_unprocessable +from trove.tests.api.instances import InstanceTestInfo +from trove.tests.api.instances import instance_info +from trove.tests.api.instances import WaitForGuestInstallationToFinish +from trove.tests.config import CONFIG +from trove.tests.util import create_dbaas_client +from trove.tests.util import test_config +from trove.tests.util.check import AttrCheck +from trove.tests.util.check import CollectionCheck +from trove.tests.util.check import TypeCheck +from trove.tests.util.mysql import create_mysql_connection +from trove.tests.util.users import Requirements +from troveclient.compat import exceptions + + +GROUP = "dbaas.api.configurations" +CONFIG_NAME = "test_configuration" +CONFIG_DESC = "configuration description" + +configuration_default = None +configuration_info = None +configuration_href = None +configuration_instance = InstanceTestInfo() +configuration_instance_id = None +sql_variables = [ + 'key_buffer_size', + 'connect_timeout', + 'join_buffer_size', +] + + +# helper methods to validate configuration is applied to instance +def _execute_query(host, user_name, password, query): + print(host, user_name, password, query) + with create_mysql_connection(host, user_name, password) as db: + result = db.execute(query) + return result + assert_true(False, "something went wrong in the sql connection") + + +def _get_address(instance_id): + result = instance_info.dbaas_admin.mgmt.instances.show(instance_id) + return result.ip[0] + + +def _test_configuration_is_applied_to_instance(instance, configuration_id): + if CONFIG.fake_mode: + raise SkipTest("configuration from sql does not work in fake mode") + instance_test = instance_info.dbaas.instances.get(instance.id) + assert_equal(configuration_id, instance_test.configuration['id']) + if configuration_id: + testconfig_info = instance_info.dbaas.configurations.get( + configuration_id) + else: + testconfig_info = instance_info.dbaas.instance.configuration( + instance.id) + testconfig_info['configuration'] + conf_instances = instance_info.dbaas.configurations.instances( + configuration_id) + config_instance_ids = [inst.id for inst in conf_instances] + assert_true(instance_test.id in config_instance_ids) + cfg_names = testconfig_info.values.keys() + + host = _get_address(instance.id) + for user in instance.users: + username = user['name'] + password = user['password'] + concat_variables = "','".join(cfg_names) + query = ("show variables where Variable_name " + "in ('%s');" % concat_variables) + actual_values = _execute_query(host, username, password, query) + print("actual_values %s" % actual_values) + print("testconfig_info.values %s" % testconfig_info.values) + assert_true(len(actual_values) == len(cfg_names)) + + # check the configs exist + attrcheck = AttrCheck() + expected_attrs = [actual_key for actual_key, actual_value in actual_values] + attrcheck.attrs_exist(testconfig_info.values, expected_attrs, + msg="Configurations parameters") + + def _get_parameter_type(name): + instance_info.dbaas.configuration_parameters.get_parameter( + instance_info.dbaas_datastore, + instance_info.dbaas_datastore_version, + name) + resp, body = instance_info.dbaas.client.last_response + print(resp) + print(body) + return json.loads(body)['type'] + + # check the config values are correct + for key, value in actual_values: + key_type = _get_parameter_type(key) + # mysql returns 'ON' and 'OFF' for True and False respectively + if value == 'ON': + converted_key_value = (str(key), 1) + elif value == 'OFF': + converted_key_value = (str(key), 0) + else: + if key_type == 'integer': + value = int(value) + converted_key_value = (str(key), value) + print("converted_key_value: %s" % str(converted_key_value)) + assert_true(converted_key_value in testconfig_info.values.items()) + + +@test(depends_on_classes=[WaitForGuestInstallationToFinish], groups=[GROUP]) +class CreateConfigurations(object): + + @test + def test_expected_configurations_parameters(self): + """test get expected configurations parameters""" + expected_attrs = ["configuration-parameters"] + instance_info.dbaas.configuration_parameters.parameters( + instance_info.dbaas_datastore, + instance_info.dbaas_datastore_version) + resp, body = instance_info.dbaas.client.last_response + attrcheck = AttrCheck() + config_parameters_dict = json.loads(body) + attrcheck.attrs_exist(config_parameters_dict, expected_attrs, + msg="Configurations parameters") + # sanity check that a few options are in the list + config_params_list = config_parameters_dict['configuration-parameters'] + config_param_keys = [] + for param in config_params_list: + config_param_keys.append(param['name']) + expected_config_params = ['key_buffer_size', 'connect_timeout'] + # check for duplicate configuration parameters + msg = "check for duplicate configuration parameters" + assert_equal(len(config_param_keys), len(set(config_param_keys)), msg) + for expected_config_item in expected_config_params: + assert_true(expected_config_item in config_param_keys) + + @test + def test_expected_get_configuration_parameter(self): + # tests get on a single parameter to verify it has expected attributes + param = 'key_buffer_size' + expected_config_params = ['name', 'restart_required', 'max', + 'min', 'type'] + instance_info.dbaas.configuration_parameters.get_parameter( + instance_info.dbaas_datastore, + instance_info.dbaas_datastore_version, + param) + resp, body = instance_info.dbaas.client.last_response + print(resp) + print(body) + attrcheck = AttrCheck() + config_parameter_dict = json.loads(body) + print(config_parameter_dict) + attrcheck.attrs_exist(config_parameter_dict, expected_config_params, + msg="Get Configuration parameter") + assert_equal(param, config_parameter_dict['name']) + + @test + def test_configurations_create_invalid_values(self): + """test create configurations with invalid values""" + values = '{"this_is_invalid": 123}' + assert_unprocessable(instance_info.dbaas.configurations.create, + CONFIG_NAME, values, CONFIG_DESC) + + @test + def test_configurations_create_invalid_value_type(self): + """test create configuration with invalild value type""" + values = '{"key_buffer_size": "this is a string not int"}' + assert_unprocessable(instance_info.dbaas.configurations.create, + CONFIG_NAME, values, CONFIG_DESC) + + @test + def test_configurations_create_value_out_of_bounds(self): + """test create configuration with value out of bounds""" + values = '{"connect_timeout": 1000000}' + assert_unprocessable(instance_info.dbaas.configurations.create, + CONFIG_NAME, values, CONFIG_DESC) + values = '{"connect_timeout": -10}' + assert_unprocessable(instance_info.dbaas.configurations.create, + CONFIG_NAME, values, CONFIG_DESC) + + @test + def test_valid_configurations_create(self): + # create a configuration with valid parameters + values = ('{"connect_timeout": 120, "local_infile": true, ' + '"collation_server": "latin1_swedish_ci"}') + expected_values = json.loads(values) + result = instance_info.dbaas.configurations.create(CONFIG_NAME, + values, + CONFIG_DESC) + resp, body = instance_info.dbaas.client.last_response + assert_equal(resp.status, 200) + global configuration_info + configuration_info = result + assert_equal(configuration_info.name, CONFIG_NAME) + assert_equal(configuration_info.description, CONFIG_DESC) + assert_equal(configuration_info.values, expected_values) + + @test(runs_after=[test_valid_configurations_create]) + def test_appending_to_existing_configuration(self): + # test being able to update and insert new parameter name and values + # to an existing configuration + values = '{"join_buffer_size": 1048576, "connect_timeout": 60}' + instance_info.dbaas.configurations.edit(configuration_info.id, + values) + resp, body = instance_info.dbaas.client.last_response + assert_equal(resp.status, 200) + + +@test(runs_after=[CreateConfigurations], groups=[GROUP]) +class AfterConfigurationsCreation(object): + + @test + def test_assign_configuration_to_invalid_instance(self): + # test assigning to an instance that does not exist + invalid_id = "invalid-inst-id" + try: + instance_info.dbaas.instances.modify(invalid_id, + configuration_info.id) + except exceptions.NotFound: + resp, body = instance_info.dbaas.client.last_response + assert_equal(resp.status, 404) + + @test + def test_assign_configuration_to_valid_instance(self): + # test assigning a configuration to an instance + print("instance_info.id: %s" % instance_info.id) + print("configuration_info: %s" % configuration_info) + print("configuration_info.id: %s" % configuration_info.id) + config_id = configuration_info.id + instance_info.dbaas.instances.modify(instance_info.id, + configuration=config_id) + resp, body = instance_info.dbaas.client.last_response + assert_equal(resp.status, 202) + + @test(depends_on=[test_assign_configuration_to_valid_instance]) + @time_out(10) + def test_get_configuration_details_from_instance_validation(self): + # validate that the configuraiton was applied correctly to the instance + inst = instance_info.dbaas.instances.get(instance_info.id) + configuration_id = inst.configuration['id'] + assert_not_equal(None, inst.configuration['id']) + _test_configuration_is_applied_to_instance(instance_info, + configuration_id) + + @test + def test_configurations_get(self): + # test that the instance shows up on the assigned configuration + result = instance_info.dbaas.configurations.get(configuration_info.id) + assert_equal(configuration_info.id, result.id) + assert_equal(configuration_info.name, result.name) + assert_equal(configuration_info.description, result.description) + + # check the result field types + with TypeCheck("configuration", result) as check: + check.has_field("id", basestring) + check.has_field("name", basestring) + check.has_field("description", basestring) + check.has_field("values", dict) + + print(result.values) + with CollectionCheck("configuration_values", result.values) as check: + # check each item has the correct type according to the rules + for (item_key, item_val) in result.values.iteritems(): + print("item_key: %s" % item_key) + print("item_val: %s" % item_val) + dbaas = instance_info.dbaas + param = dbaas.configuration_parameters.get_parameter( + instance_info.dbaas_datastore, + instance_info.dbaas_datastore_version, + item_key) + if param.type == 'integer': + check.has_element(item_key, int) + if param.type == 'string': + check.has_element(item_key, basestring) + if param.type == 'boolean': + check.has_element(item_key, bool) + + # Test to make sure that another user is not able to GET this config + reqs = Requirements(is_admin=False) + test_auth_user = instance_info.user.auth_user + other_user = CONFIG.users.find_user(reqs, black_list=[test_auth_user]) + other_user_tenant_id = other_user.tenant_id + client_tenant_id = instance_info.user.tenant_id + if other_user_tenant_id == client_tenant_id: + other_user = CONFIG.users.find_user(reqs, + black_list=[ + instance_info.user.auth_user, + other_user]) + print(other_user) + print(other_user.__dict__) + other_client = create_dbaas_client(other_user) + assert_raises(exceptions.NotFound, other_client.configurations.get, + configuration_info.id) + + +@test(runs_after=[AfterConfigurationsCreation], groups=[GROUP]) +class ListConfigurations(object): + + @test + def test_configurations_list(self): + # test listing configurations show up + result = instance_info.dbaas.configurations.list() + exists = [config for config in result if + config.id == configuration_info.id] + assert_equal(1, len(exists)) + configuration = exists[0] + assert_equal(configuration.id, configuration_info.id) + assert_equal(configuration.name, configuration_info.name) + assert_equal(configuration.description, configuration_info.description) + + @test + def test_configurations_list_for_instance(self): + # test getting an instance shows the configuration assigned shows up + instance = instance_info.dbaas.instances.get(instance_info.id) + assert_equal(instance.configuration['id'], configuration_info.id) + assert_equal(instance.configuration['name'], configuration_info.name) + # expecting two things in links, href and bookmark + assert_equal(2, len(instance.configuration['links'])) + link = instance.configuration['links'][0] + global configuration_href + configuration_href = link['href'] + + @test + def test_get_default_configuration_on_instance(self): + # test the api call to get the default template of an instance exists + result = instance_info.dbaas.instances.configuration(instance_info.id) + global configuration_default + configuration_default = result + assert_not_equal(None, result.configuration) + + @test + def test_changing_configuration_with_nondynamic_parameter(self): + # test that changing a non-dynamic parameter is applied to instance + # and show that the instance requires a restart + values = ('{"join_buffer_size":1048576,' + '"innodb_buffer_pool_size":57671680}') + instance_info.dbaas.configurations.update(configuration_info.id, + values) + resp, body = instance_info.dbaas.client.last_response + assert_equal(resp.status, 202) + + instance_info.dbaas.configurations.get(configuration_info.id) + resp, body = instance_info.dbaas.client.last_response + assert_equal(resp.status, 200) + + @test(depends_on=[test_changing_configuration_with_nondynamic_parameter]) + @time_out(20) + def test_waiting_for_instance_in_restart_required(self): + def result_is_not_active(): + instance = instance_info.dbaas.instances.get( + instance_info.id) + if instance.status == "ACTIVE": + return False + else: + return True + poll_until(result_is_not_active) + + instance = instance_info.dbaas.instances.get(instance_info.id) + resp, body = instance_info.dbaas.client.last_response + assert_equal(resp.status, 200) + print(instance.status) + assert_equal('RESTART_REQUIRED', instance.status) + + @test(depends_on=[test_waiting_for_instance_in_restart_required]) + def test_restart_service_should_return_active(self): + # test that after restarting the instance it becomes active + instance_info.dbaas.instances.restart(instance_info.id) + resp, body = instance_info.dbaas.client.last_response + assert_equal(resp.status, 202) + + def result_is_active(): + instance = instance_info.dbaas.instances.get( + instance_info.id) + if instance.status == "ACTIVE": + return True + else: + assert_equal("REBOOT", instance.status) + return False + poll_until(result_is_active) + + @test(depends_on=[test_restart_service_should_return_active]) + @time_out(10) + def test_get_configuration_details_from_instance_validation(self): + # validate that the configuraiton was applied correctly to the instance + inst = instance_info.dbaas.instances.get(instance_info.id) + configuration_id = inst.configuration['id'] + assert_not_equal(None, inst.configuration['id']) + _test_configuration_is_applied_to_instance(instance_info, + configuration_id) + + +@test(runs_after=[ListConfigurations], groups=[GROUP]) +class StartInstanceWithConfiguration(object): + + @test + def test_start_instance_with_configuration(self): + # test that a new instance will apply the configuration on create + if test_config.auth_strategy == "fake": + raise SkipTest("Skipping instance start with configuration " + "test for fake mode.") + global configuration_instance + databases = [] + databases.append({"name": "firstdbconfig", "character_set": "latin2", + "collate": "latin2_general_ci"}) + databases.append({"name": "db2"}) + configuration_instance.databases = databases + users = [] + users.append({"name": "liteconf", "password": "liteconfpass", + "databases": [{"name": "firstdbconfig"}]}) + configuration_instance.users = users + configuration_instance.name = "TEST_" + str(datetime.now()) + "_config" + flavor_href = instance_info.dbaas_flavor_href + configuration_instance.dbaas_flavor_href = flavor_href + configuration_instance.volume = instance_info.volume + + result = instance_info.dbaas.instances.create( + configuration_instance.name, + configuration_instance.dbaas_flavor_href, + configuration_instance.volume, + configuration_instance.databases, + configuration_instance.users, + availability_zone="nova", + configuration=configuration_href) + assert_equal(200, instance_info.dbaas.last_http_code) + assert_equal("BUILD", result.status) + configuration_instance.id = result.id + + +@test(runs_after=[StartInstanceWithConfiguration], groups=[GROUP]) +class WaitForConfigurationInstanceToFinish(object): + + @test + @time_out(60 * 7) + def test_instance_with_configuration_active(self): + # wait for the instance to become active + if test_config.auth_strategy == "fake": + raise SkipTest("Skipping instance start with configuration " + "test for fake mode.") + + def result_is_active(): + instance = instance_info.dbaas.instances.get( + configuration_instance.id) + if instance.status == "ACTIVE": + return True + else: + assert_equal("BUILD", instance.status) + return False + + poll_until(result_is_active) + + @test(depends_on=[test_instance_with_configuration_active]) + @time_out(10) + def test_get_configuration_details_from_instance_validation(self): + # validate that the configuraiton was applied correctly to the instance + inst = instance_info.dbaas.instances.get(configuration_instance.id) + configuration_id = inst.configuration['id'] + assert_not_equal(None, inst.configuration['id']) + _test_configuration_is_applied_to_instance(configuration_instance, + configuration_id) + + +@test(runs_after=[WaitForConfigurationInstanceToFinish], groups=[GROUP]) +class DeleteConfigurations(object): + + @test + def test_delete_invalid_configuration_not_found(self): + # test deleting a configuration that does not exist throws exception + invalid_configuration_id = "invalid-config-id" + assert_raises(exceptions.NotFound, + instance_info.dbaas.configurations.delete, + invalid_configuration_id) + + @test + def test_unable_delete_instance_configurations(self): + # test deleting a configuration that is assigned to + # an instance is not allowed. + assert_raises(exceptions.BadRequest, + instance_info.dbaas.configurations.delete, + configuration_info.id) + + @test(depends_on=[test_unable_delete_instance_configurations]) + @time_out(30) + def test_unassign_configuration_from_instances(self): + # test to unassign configuration from instance + instance_info.dbaas.instances.modify(configuration_instance.id, + configuration="") + resp, body = instance_info.dbaas.client.last_response + assert_equal(resp.status, 202) + instance_info.dbaas.instances.get(configuration_instance.id) + #test that config group is not removed + instance_info.dbaas.instances.modify(instance_info.id, + configuration=None) + resp, body = instance_info.dbaas.client.last_response + assert_equal(resp.status, 202) + instance_info.dbaas.instances.get(instance_info.id) + + def result_has_no_configuration(): + instance = instance_info.dbaas.instances.get(inst_info.id) + if hasattr(instance, 'configuration'): + return False + else: + return True + inst_info = instance_info + poll_until(result_has_no_configuration) + inst_info = configuration_instance + poll_until(result_has_no_configuration) + + @test(depends_on=[test_unassign_configuration_from_instances]) + def test_no_instances_on_configuration(self): + # test there is no configuration on the instance after unassigning + result = instance_info.dbaas.configurations.get(configuration_info.id) + assert_equal(configuration_info.id, result.id) + assert_equal(configuration_info.name, result.name) + assert_equal(configuration_info.description, result.description) + print(configuration_instance.id) + print(instance_info.id) + + @test(depends_on=[test_no_instances_on_configuration]) + def test_delete_unassigned_configuration(self): + # test that we can delete the configuration after no instances are + # assigned to it any longer + instance_info.dbaas.configurations.delete(configuration_info.id) + resp, body = instance_info.dbaas.client.last_response + assert_equal(resp.status, 202) + + @test(depends_on=[test_unassign_configuration_from_instances]) + @time_out(120) + def test_restart_service_after_unassign_return_active(self): + def result_is_not_active(): + instance = instance_info.dbaas.instances.get( + instance_info.id) + if instance.status == "ACTIVE": + return False + else: + return True + poll_until(result_is_not_active) + + config = instance_info.dbaas.configurations.list() + print(config) + instance = instance_info.dbaas.instances.get(instance_info.id) + print(instance.__dict__) + resp, body = instance_info.dbaas.client.last_response + assert_equal(resp.status, 200) + print(instance.status) + assert_equal('RESTART_REQUIRED', instance.status) + + @test(depends_on=[test_restart_service_after_unassign_return_active]) + @time_out(120) + def test_restart_service_should_return_active(self): + # test that after restarting the instance it becomes active + instance_info.dbaas.instances.restart(instance_info.id) + resp, body = instance_info.dbaas.client.last_response + assert_equal(resp.status, 202) + + def result_is_active(): + instance = instance_info.dbaas.instances.get( + instance_info.id) + if instance.status == "ACTIVE": + return True + else: + assert_equal("REBOOT", instance.status) + return False + poll_until(result_is_active) + + @test(depends_on=[test_delete_unassigned_configuration]) + @time_out(120) + def test_delete_configuration_instance(self): + # test that we can delete the instance even though there is a + # configuration applied to the instance + instance_info.dbaas.instances.delete(configuration_instance.id) + assert_equal(202, instance_info.dbaas.last_http_code) + + def instance_is_gone(): + try: + instance_info.dbaas.instances.get(configuration_instance.id) + return False + except exceptions.NotFound: + return True + + poll_until(instance_is_gone) + assert_raises(exceptions.NotFound, instance_info.dbaas.instances.get, + configuration_instance.id) diff --git a/trove/tests/api/instances.py b/trove/tests/api/instances.py index 1bf9a4704d..70a5e61a3d 100644 --- a/trove/tests/api/instances.py +++ b/trove/tests/api/instances.py @@ -350,6 +350,7 @@ class CreateInstance(object): result = instance_info.initial_result instance_info.id = result.id + instance_info.dbaas_datastore_version = result.datastore['version'] report = CONFIG.get_report() report.log("Instance UUID = %s" % instance_info.id) diff --git a/trove/tests/fakes/guestagent.py b/trove/tests/fakes/guestagent.py index 875b8a9829..be1ae13b0b 100644 --- a/trove/tests/fakes/guestagent.py +++ b/trove/tests/fakes/guestagent.py @@ -37,6 +37,7 @@ class FakeGuest(object): self.root_was_enabled = False self.version = 1 self.grants = {} + self.overrides = {} # Our default admin user. self._create_user({ @@ -209,7 +210,7 @@ class FakeGuest(object): def prepare(self, memory_mb, packages, databases, users, device_path=None, mount_point=None, backup_info=None, config_contents=None, - root_password=None): + root_password=None, overrides=None): from trove.instance.models import DBInstance from trove.instance.models import InstanceServiceStatus from trove.guestagent.models import AgentHeartBeat @@ -218,6 +219,7 @@ class FakeGuest(object): instance_name = DBInstance.find_by(id=self.id).name self.create_user(users) self.create_database(databases) + self.overrides = overrides or {} def update_db(): status = InstanceServiceStatus.find_by(instance_id=self.id) @@ -317,6 +319,12 @@ class FakeGuest(object): def resize_fs(self, device_path=None, mount_point=None): pass + def update_overrides(self, overrides, remove=False): + self.overrides = overrides + + def apply_overrides(self, overrides): + self.overrides = overrides + def get_or_create(id): if id not in DB: diff --git a/trove/tests/unittests/configuration/__init__.py b/trove/tests/unittests/configuration/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/trove/tests/unittests/configuration/test_configuration_controller.py b/trove/tests/unittests/configuration/test_configuration_controller.py new file mode 100644 index 0000000000..048c691fe6 --- /dev/null +++ b/trove/tests/unittests/configuration/test_configuration_controller.py @@ -0,0 +1,102 @@ +# Copyright 2014 Rackspace +# 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. +# + +import jsonschema +from testtools import TestCase +from trove.configuration.service import ConfigurationsController + + +class TestConfigurationController(TestCase): + def setUp(self): + super(TestConfigurationController, self).setUp() + self.controller = ConfigurationsController() + + def test_validate_create_configuration(self): + body = { + "configuration": { + "values": {}, + "name": "test", + "datastore": { + "type": "test_type", + "version": "test_version" + } + } + } + schema = self.controller.get_schema('create', body) + self.assertIsNotNone(schema) + validator = jsonschema.Draft4Validator(schema) + self.assertTrue(validator.is_valid(body)) + + def test_validate_create_configuration_no_datastore(self): + body = { + "configuration": { + "values": {}, + "name": "test" + } + } + schema = self.controller.get_schema('create', body) + self.assertIsNotNone(schema) + validator = jsonschema.Draft4Validator(schema) + self.assertTrue(validator.is_valid(body)) + + def test_validate_create_invalid_values_param(self): + body = { + "configuration": { + "values": '', + "name": "test", + "datastore": { + "type": "test_type", + "version": "test_version" + } + } + } + schema = self.controller.get_schema('create', body) + self.assertIsNotNone(schema) + validator = jsonschema.Draft4Validator(schema) + self.assertFalse(validator.is_valid(body)) + errors = sorted(validator.iter_errors(body), key=lambda e: e.path) + self.assertEqual(errors[0].message, + "'' is not of type 'object'") + + def test_validate_create_invalid_name_param(self): + body = { + "configuration": { + "values": {}, + "name": "", + "datastore": { + "type": "test_type", + "version": "test_version" + } + } + } + schema = self.controller.get_schema('create', body) + self.assertIsNotNone(schema) + validator = jsonschema.Draft4Validator(schema) + self.assertFalse(validator.is_valid(body)) + errors = sorted(validator.iter_errors(body), key=lambda e: e.path) + self.assertEqual(errors[0].message, + "'' is too short") + + def test_validate_edit_configuration(self): + body = { + "configuration": { + "values": {} + } + } + schema = self.controller.get_schema('edit', body) + self.assertIsNotNone(schema) + validator = jsonschema.Draft4Validator(schema) + self.assertTrue(validator.is_valid(body)) diff --git a/trove/tests/unittests/datastore/__init__.py b/trove/tests/unittests/datastore/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/trove/tests/unittests/datastore/test_datastore.py b/trove/tests/unittests/datastore/test_datastore.py new file mode 100644 index 0000000000..8a77cfbe8d --- /dev/null +++ b/trove/tests/unittests/datastore/test_datastore.py @@ -0,0 +1,28 @@ +# Copyright 2014 Rackspace +# 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. + +from testtools import TestCase +from trove.datastore import models as datastore_models +from trove.common.exception import DatastoreDefaultDatastoreNotFound + + +class TestDatastore(TestCase): + def setUp(self): + super(TestDatastore, self).setUp() + + def test_create_failure_with_datastore_default_notfound(self): + self.assertRaises( + DatastoreDefaultDatastoreNotFound, + datastore_models.get_datastore_version) diff --git a/trove/tests/unittests/guestagent/test_api.py b/trove/tests/unittests/guestagent/test_api.py index ec25d3b184..a2c7c99324 100644 --- a/trove/tests/unittests/guestagent/test_api.py +++ b/trove/tests/unittests/guestagent/test_api.py @@ -242,6 +242,18 @@ class ApiTest(testtools.TestCase): self.api.create_backup({'id': '123'}) self._verify_rpc_cast(exp_msg) + def test_update_overrides(self): + exp_msg = RpcMsgMatcher('update_overrides', 'overrides', 'remove') + self._mock_rpc_cast(exp_msg) + self.api.update_overrides('123') + self._verify_rpc_cast(exp_msg) + + def test_apply_overrides(self): + exp_msg = RpcMsgMatcher('apply_overrides', 'overrides') + self._mock_rpc_cast(exp_msg) + self.api.apply_overrides('123') + self._verify_rpc_cast(exp_msg) + def _verify_rpc_connection_and_cast(self, rpc, mock_conn, exp_msg): verify(rpc).create_connection(new=True) verify(mock_conn).create_consumer(self.api._get_routing_key(), None, @@ -255,12 +267,13 @@ class ApiTest(testtools.TestCase): exp_msg = RpcMsgMatcher('prepare', 'memory_mb', 'packages', 'databases', 'users', 'device_path', 'mount_point', 'backup_info', - 'config_contents', 'root_password') - + 'config_contents', 'root_password', + 'overrides') when(rpc).cast(any(), any(), exp_msg).thenReturn(None) self.api.prepare('2048', 'package1', 'db1', 'user1', '/dev/vdt', - '/mnt/opt', 'bkup-1232', 'cont', '1-2-3-4') + '/mnt/opt', 'bkup-1232', 'cont', '1-2-3-4', + 'override') self._verify_rpc_connection_and_cast(rpc, mock_conn, exp_msg) @@ -271,13 +284,14 @@ class ApiTest(testtools.TestCase): exp_msg = RpcMsgMatcher('prepare', 'memory_mb', 'packages', 'databases', 'users', 'device_path', 'mount_point', 'backup_info', - 'config_contents', 'root_password') + 'config_contents', 'root_password', + 'overrides') when(rpc).cast(any(), any(), exp_msg).thenReturn(None) bkup = {'id': 'backup_id_123'} self.api.prepare('2048', 'package1', 'db1', 'user1', '/dev/vdt', - '/mnt/opt', bkup, 'cont', '1-2-3-4') - + '/mnt/opt', bkup, 'cont', '1-2-3-4', + 'overrides') self._verify_rpc_connection_and_cast(rpc, mock_conn, exp_msg) def test_upgrade(self): diff --git a/trove/tests/unittests/guestagent/test_dbaas.py b/trove/tests/unittests/guestagent/test_dbaas.py index 67946aed07..5b8269c21e 100644 --- a/trove/tests/unittests/guestagent/test_dbaas.py +++ b/trove/tests/unittests/guestagent/test_dbaas.py @@ -658,6 +658,13 @@ class MySqlAppTest(testtools.TestCase): self.mySqlApp.start_db_with_conf_changes, Mock()) + def test_remove_overrides(self): + + from trove.common.exception import ProcessExecutionError + mocked = Mock(side_effect=ProcessExecutionError('Error')) + dbaas.utils.execute_with_timeout = mocked + self.assertRaises(ProcessExecutionError, self.mySqlApp.start_mysql) + class MySqlAppInstallTest(MySqlAppTest): @@ -694,7 +701,7 @@ class MySqlAppInstallTest(MySqlAppTest): self.mysql_starts_successfully() sqlalchemy.create_engine = Mock() - self.mySqlApp.secure('contents') + self.mySqlApp.secure('contents', None) self.assertTrue(self.mySqlApp.stop_db.called) self.assertTrue(self.mySqlApp._write_mycnf.called) @@ -728,7 +735,7 @@ class MySqlAppInstallTest(MySqlAppTest): self.mysql_starts_successfully() sqlalchemy.create_engine = Mock() - self.assertRaises(IOError, self.mySqlApp.secure, "foo") + self.assertRaises(IOError, self.mySqlApp.secure, "foo", None) self.assertTrue(self.mySqlApp.stop_db.called) self.assertTrue(self.mySqlApp._write_mycnf.called) @@ -789,7 +796,7 @@ class MySqlAppMockTest(testtools.TestCase): any(), any(), any()).thenReturn(True) app = MySqlApp(mock_status) when(dbaas).clear_expired_password().thenReturn(None) - self.assertRaises(TypeError, app.secure, None) + self.assertRaises(TypeError, app.secure, None, None) verify(mock_conn, atleast=2).execute(any()) inorder.verify(mock_status).wait_for_real_status_to_change_to( @@ -814,7 +821,7 @@ class MySqlAppMockTest(testtools.TestCase): when(app)._write_mycnf(any(), any()).thenReturn(True) when(app).start_mysql().thenReturn(None) when(app).stop_db().thenReturn(None) - app.secure('foo') + app.secure('foo', None) verify(mock_conn, never).execute(TextClauseMatcher('root')) diff --git a/trove/tests/unittests/guestagent/test_manager.py b/trove/tests/unittests/guestagent/test_manager.py index c61ca1a6b7..10db558d9e 100644 --- a/trove/tests/unittests/guestagent/test_manager.py +++ b/trove/tests/unittests/guestagent/test_manager.py @@ -163,7 +163,8 @@ class GuestAgentManagerTest(testtools.TestCase): is_root_enabled=True) def _prepare_dynamic(self, device_path='/dev/vdb', is_mysql_installed=True, - backup_id=None, is_root_enabled=False): + backup_id=None, is_root_enabled=False, + overrides=None): # covering all outcomes is starting to cause trouble here COUNT = 1 if device_path else 0 @@ -205,7 +206,8 @@ class GuestAgentManagerTest(testtools.TestCase): users=None, device_path=device_path, mount_point='/var/lib/mysql', - backup_info=backup_info) + backup_info=backup_info, + overrides=overrides) # verification/assertion verify(mock_status).begin_install() @@ -217,7 +219,7 @@ class GuestAgentManagerTest(testtools.TestCase): verify(backup).restore(self.context, backup_info, '/var/lib/mysql') verify(dbaas.MySqlApp).install_if_needed(any()) # We dont need to make sure the exact contents are there - verify(dbaas.MySqlApp).secure(any()) + verify(dbaas.MySqlApp).secure(any(), overrides) verify(dbaas.MySqlAdmin, never).create_database() verify(dbaas.MySqlAdmin, never).create_user() verify(dbaas.MySqlApp).secure_root(secure_remote_root=any()) diff --git a/trove/tests/unittests/instance/test_instance_views.py b/trove/tests/unittests/instance/test_instance_views.py index fc778f63cb..371a28a7fd 100644 --- a/trove/tests/unittests/instance/test_instance_views.py +++ b/trove/tests/unittests/instance/test_instance_views.py @@ -44,8 +44,10 @@ class InstanceDetailViewTest(TestCase): super(InstanceDetailViewTest, self).setUp() self.build_links_method = InstanceView._build_links self.build_flavor_links_method = InstanceView._build_flavor_links + self.build_config_method = InstanceDetailView._build_configuration_info InstanceView._build_links = Mock() InstanceView._build_flavor_links = Mock() + InstanceDetailView._build_configuration_info = Mock() self.instance = Mock() self.instance.created = 'Yesterday' self.instance.updated = 'Now' @@ -62,6 +64,7 @@ class InstanceDetailViewTest(TestCase): super(InstanceDetailViewTest, self).tearDown() InstanceView._build_links = self.build_links_method InstanceView._build_flavor_links = self.build_flavor_links_method + InstanceDetailView._build_configuration_info = self.build_config_method def test_data_hostname(self): view = InstanceDetailView(self.instance, Mock()) diff --git a/trove/tests/util/__init__.py b/trove/tests/util/__init__.py index 81ee908a6b..c3a3fb7c30 100644 --- a/trove/tests/util/__init__.py +++ b/trove/tests/util/__init__.py @@ -235,6 +235,7 @@ def iso_time(time_string): # https://bugs.launchpad.net/trove-integration/+bug/1228306 +#TODO(cp16net): DO NOT USE needs to be removed def mysql_connection(): cls = CONFIG.get('mysql_connection', "local.MySqlConnection")