diff --git a/.zuul.yaml b/.zuul.yaml index 143b9a5cb0..f5ba550ca0 100644 --- a/.zuul.yaml +++ b/.zuul.yaml @@ -14,14 +14,16 @@ - openstack-tox-pylint - trove-tox-bandit-baseline: voting: false - - trove-tempest + - trove-tempest: + voting: false - trove-tempest-ipv6-only: voting: false gate: queue: trove jobs: - openstack-tox-pylint - - trove-tempest + - trove-tempest: + voting: false experimental: jobs: - trove-functional-mysql diff --git a/doc/source/user/index.rst b/doc/source/user/index.rst index d127d64f1a..c26a3af4c1 100644 --- a/doc/source/user/index.rst +++ b/doc/source/user/index.rst @@ -17,6 +17,6 @@ handling complex administrative tasks. backup-db-incremental.rst manage-db-config.rst set-up-replication.rst - set-up-clustering.rst upgrade-datastore.rst + set-up-clustering.rst upgrade-cluster-datastore.rst diff --git a/doc/source/user/upgrade-cluster-datastore.rst b/doc/source/user/upgrade-cluster-datastore.rst index 08f513758b..6fe946c53d 100644 --- a/doc/source/user/upgrade-cluster-datastore.rst +++ b/doc/source/user/upgrade-cluster-datastore.rst @@ -2,6 +2,11 @@ Upgrade cluster datastore ========================= +.. caution:: + + Database clustering function is still in experimental, should not be used + in production environment. + Upgrading datastore for cluster instances is very similar to upgrading a single instance. diff --git a/doc/source/user/upgrade-datastore.rst b/doc/source/user/upgrade-datastore.rst index 1a3e69f650..3947b37cd0 100644 --- a/doc/source/user/upgrade-datastore.rst +++ b/doc/source/user/upgrade-datastore.rst @@ -8,58 +8,71 @@ configuration files of your database. To perform datastore upgrade, you need: +- A Trove database instance to be upgrade. - A guest image with the target datastore version. -- A Trove database instance to be upgrade. +This guide shows you how to upgrade MySQL datastore from 5.7.29 to 5.7.30 for a +database instance. -This example shows you how to upgrade Redis datastore (version 3.2.6) -for a single instance database. +.. warning:: -.. note:: - - **Before** upgrading, make sure that: - - - Your target datastore is binary compatible with the current - datastore. Each database provider has its own compatibilty - policy. Usually there shouldn't be any problem when - performing an upgrade within minor versions. - - - You **do not** downgrade your datastore. - - - Target versions is supported by Trove. For instance, Trove - doesn't support Cassandra >=2.2 at this moment so you - shouldn't perform an upgrade from 2.1 to 2.2. + Datastore upgrade could cause downtime of the database service. Upgrading datastore ~~~~~~~~~~~~~~~~~~~ -#. **Check instance status** +#. **Check datastore versions in the system** + + In my environment, both datastore version 5.7.29 and 5.7.30 are defined for + MySQL. + + .. code-block:: console + + $ openstack datastore list + +--------------------------------------+-------+ + | ID | Name | + +--------------------------------------+-------+ + | 50bed39d-6788-4a0d-8d74-321012bb6b55 | mysql | + +--------------------------------------+-------+ + $ openstack datastore version list mysql + +--------------------------------------+--------+ + | ID | Name | + +--------------------------------------+--------+ + | 70c68d0a-27e1-4fbd-bd3b-f29d42ce1a7d | 5.7.29 | + | cf91aa9a-2192-4ec4-b7ce-5cac3b1e7dbe | 5.7.30 | + +--------------------------------------+--------+ + +#. **Create a new instance with datastore version 5.7.29** Make sure the instance status is HEALTHY before upgrading. .. code-block:: console + $ openstack database instance create test-mysql-upgrade \ + d2 \ + --size 1 \ + --nic net-id=$netid \ + --datastore mysql --datastore_version 5.7.29 \ + --databases testdb --users user:password $ openstack database instance list - +--------------------------------------+------------+-----------+-------------------+---------+-----------+-----------+------+-----------+ - | ID | Name | Datastore | Datastore Version | Status | Addresses | Flavor ID | Size | Region | - +--------------------------------------+------------+-----------+-------------------+---------+-----------+-----------+------+-----------+ - | 55411e95-1670-497f-8d92-0179f3b4fdd4 | redis_test | redis | 3.2.6 | HEALTHY | 10.1.0.25 | 6 | 1 | RegionOne | - +--------------------------------------+------------+-----------+-------------------+---------+-----------+-----------+------+-----------+ + +--------------------------------------+--------------------+-----------+-------------------+---------+-----------------------------------------------+-----------+------+-----------+---------+ + | ID | Name | Datastore | Datastore Version | Status | Addresses | Flavor ID | Size | Region | Role | + +--------------------------------------+--------------------+-----------+-------------------+---------+-----------------------------------------------+-----------+------+-----------+---------+ + | 32eb56b0-d10d-43e9-b59e-1e4b0979e5dd | test-mysql-upgrade | mysql | 5.7.29 | HEALTHY | [{'address': '10.0.0.54', 'type': 'private'}] | d2 | 1 | RegionOne | | + +--------------------------------------+--------------------+-----------+-------------------+---------+-----------------------------------------------+-----------+------+-----------+---------+ -#. **Check if target version is available** - - Use :command:`openstack datastore version list` command to list - all available versions your datastore. + Check the MySQL version by connecting with the database: .. code-block:: console - $ openstack datastore version list redis - +--------------------------------------+-------+ - | ID | Name | - +--------------------------------------+-------+ - | 483debec-b7c3-4167-ab1d-1765795ed7eb | 3.2.6 | - | 507f666e-193c-4194-9d9d-da8342dcb4f1 | 3.2.7 | - +--------------------------------------+-------+ + $ ip=10.0.0.54 + $ mysql -u user -ppassword -h $ip testdb + mysql> SELECT @@GLOBAL.innodb_version; + +-------------------------+ + | @@GLOBAL.innodb_version | + +-------------------------+ + | 5.7.29 | + +-------------------------+ #. **Run upgrade** @@ -68,7 +81,7 @@ Upgrading datastore .. code-block:: console - $ openstack database instance upgrade 55411e95-1670-497f-8d92-0179f3b4fdd4 3.2.7 + $ openstack database instance upgrade 32eb56b0-d10d-43e9-b59e-1e4b0979e5dd cf91aa9a-2192-4ec4-b7ce-5cac3b1e7dbe #. **Wait until status changes from UPGRADE to HEALTHY** @@ -78,24 +91,26 @@ Upgrading datastore .. code-block:: console $ openstack database instance list - +--------------------------------------+------------+-----------+-------------------+---------+-----------+-----------+------+-----------+ - | ID | Name | Datastore | Datastore Version | Status | Addresses | Flavor ID | Size | Region | - +--------------------------------------+------------+-----------+-------------------+---------+-----------+-----------+------+-----------+ - | 55411e95-1670-497f-8d92-0179f3b4fdd4 | redis_test | redis | 3.2.7 | UPGRADE | 10.1.0.25 | 6 | 5 | RegionOne | - +--------------------------------------+------------+-----------+-------------------+---------+-----------+-----------+------+-----------+ + +--------------------------------------+--------------------+-----------+-------------------+---------+-----------------------------------------------+-----------+------+-----------+---------+ + | ID | Name | Datastore | Datastore Version | Status | Addresses | Flavor ID | Size | Region | Role | + +--------------------------------------+--------------------+-----------+-------------------+---------+-----------------------------------------------+-----------+------+-----------+---------+ + | 32eb56b0-d10d-43e9-b59e-1e4b0979e5dd | test-mysql-upgrade | mysql | 5.7.30 | UPGRADE | [{'address': '10.0.0.54', 'type': 'private'}] | d2 | 1 | RegionOne | | + +--------------------------------------+--------------------+-----------+-------------------+---------+-----------------------------------------------+-----------+------+-----------+---------+ $ openstack database instance list - +--------------------------------------+------------+-----------+-------------------+---------+-----------+-----------+------+-----------+ - | ID | Name | Datastore | Datastore Version | Status | Addresses | Flavor ID | Size | Region | - +--------------------------------------+------------+-----------+-------------------+---------+-----------+-----------+------+-----------+ - | 55411e95-1670-497f-8d92-0179f3b4fdd4 | redis_test | redis | 3.2.7 | HEALTHY | 10.1.0.25 | 6 | 5 | RegionOne | - +--------------------------------------+------------+-----------+-------------------+---------+-----------+-----------+------+-----------+ + +--------------------------------------+--------------------+-----------+-------------------+---------+-----------------------------------------------+-----------+------+-----------+---------+ + | ID | Name | Datastore | Datastore Version | Status | Addresses | Flavor ID | Size | Region | Role | + +--------------------------------------+--------------------+-----------+-------------------+---------+-----------------------------------------------+-----------+------+-----------+---------+ + | 32eb56b0-d10d-43e9-b59e-1e4b0979e5dd | test-mysql-upgrade | mysql | 5.7.30 | HEALTHY | [{'address': '10.0.0.54', 'type': 'private'}] | d2 | 1 | RegionOne | | + +--------------------------------------+--------------------+-----------+-------------------+---------+-----------------------------------------------+-----------+------+-----------+---------+ -Other datastores -~~~~~~~~~~~~~~~~ + Check the MySQL version again: -Upgrade for other datastores works in the same way. Currently Trove -supports upgrades for the following datastores: + .. code-block:: console -- MySQL -- MariaDB -- Redis + $ mysql -u user -ppassword -h $ip testdb + mysql> SELECT @@GLOBAL.innodb_version; + +-------------------------+ + | @@GLOBAL.innodb_version | + +-------------------------+ + | 5.7.30 | + +-------------------------+ diff --git a/releasenotes/notes/victoria-expired-database-status.yaml b/releasenotes/notes/victoria-expired-database-status.yaml new file mode 100644 index 0000000000..2c46f8ef7a --- /dev/null +++ b/releasenotes/notes/victoria-expired-database-status.yaml @@ -0,0 +1,4 @@ +--- +fixes: + - When the trove-guestagent failed to update the datastore service status, + the instance status should be ERROR. diff --git a/tools/trove-pylint.config b/tools/trove-pylint.config index 85571893be..98251bfa24 100644 --- a/tools/trove-pylint.config +++ b/tools/trove-pylint.config @@ -1239,6 +1239,12 @@ "Instance of 'DBInstance' has no 'encrypted_key' member", "DBInstance.key" ], + [ + "trove/instance/models.py", + "E1101", + "Instance of 'InstanceServiceStatus' has no 'updated_at' member", + "InstanceServiceStatus.is_uptodate" + ], [ "trove/instance/models.py", "no-member", @@ -1323,6 +1329,12 @@ "Instance of 'DBInstance' has no 'encrypted_key' member", "DBInstance.key" ], + [ + "trove/instance/models.py", + "no-member", + "Instance of 'InstanceServiceStatus' has no 'updated_at' member", + "InstanceServiceStatus.is_uptodate" + ], [ "trove/instance/service.py", "E1101", diff --git a/trove/common/strategies/cluster/experimental/mongodb/taskmanager.py b/trove/common/strategies/cluster/experimental/mongodb/taskmanager.py index 7e680d9d69..56a80afd7c 100644 --- a/trove/common/strategies/cluster/experimental/mongodb/taskmanager.py +++ b/trove/common/strategies/cluster/experimental/mongodb/taskmanager.py @@ -17,17 +17,16 @@ from eventlet.timeout import Timeout from oslo_log import log as logging from trove.common import cfg -from trove.common.exception import PollTimeOut -from trove.common.instance import ServiceStatuses -from trove.common.strategies.cluster import base from trove.common import utils +from trove.common.exception import PollTimeOut +from trove.common.strategies.cluster import base from trove.instance import models +from trove.instance import tasks as inst_tasks from trove.instance.models import DBInstance from trove.instance.models import Instance -from trove.instance import tasks as inst_tasks +from trove.instance.service_status import ServiceStatuses from trove.taskmanager import api as task_api -import trove.taskmanager.models as task_models - +from trove.taskmanager import models as task_models LOG = logging.getLogger(__name__) CONF = cfg.CONF diff --git a/trove/conductor/manager.py b/trove/conductor/manager.py index a09d485ffb..22fb75c578 100644 --- a/trove/conductor/manager.py +++ b/trove/conductor/manager.py @@ -19,12 +19,12 @@ from oslo_service import periodic_task from trove.backup import models as bkup_models from trove.common import cfg from trove.common import exception as trove_exception -from trove.common.instance import ServiceStatus from trove.common.rpc import version as rpc_version from trove.common.serializable_notification import SerializableNotification from trove.conductor.models import LastSeen from trove.extensions.mysql import models as mysql_models from trove.instance import models as inst_models +from trove.instance.service_status import ServiceStatus LOG = logging.getLogger(__name__) CONF = cfg.CONF @@ -89,8 +89,8 @@ class Manager(periodic_task.PeriodicTasks): if self._message_too_old(instance_id, 'heartbeat', sent): return if payload.get('service_status') is not None: - status.set_status(ServiceStatus.from_description( - payload['service_status'])) + status.set_status( + ServiceStatus.from_description(payload['service_status'])) status.save() def update_backup(self, context, instance_id, backup_id, diff --git a/trove/guestagent/api.py b/trove/guestagent/api.py index 27e626e0cb..67c3baa544 100644 --- a/trove/guestagent/api.py +++ b/trove/guestagent/api.py @@ -380,6 +380,14 @@ class API(object): self.agent_high_timeout, version=version, upgrade_info=upgrade_info) + def upgrade(self, upgrade_info): + """Upgrade database service.""" + LOG.debug("Sending the call to upgrade database service.") + version = self.API_BASE_VERSION + + return self._cast("upgrade", version=version, + upgrade_info=upgrade_info) + def restart(self): """Restart the database server.""" LOG.debug("Sending the call to restart the database process " @@ -419,16 +427,6 @@ class API(object): self._call("stop_db", self.agent_low_timeout, version=version) - def upgrade(self, instance_version, location, metadata=None): - """Make an asynchronous call to self upgrade the guest agent.""" - LOG.debug("Sending an upgrade call to nova-guest.") - version = self.API_BASE_VERSION - - self._cast("upgrade", version=version, - instance_version=instance_version, - location=location, - metadata=metadata) - def get_volume_info(self): """Make a synchronous call to get volume info for the container.""" LOG.debug("Check Volume Info on instance %s.", self.id) diff --git a/trove/guestagent/datastore/manager.py b/trove/guestagent/datastore/manager.py index 0a639417c2..a7e126f124 100644 --- a/trove/guestagent/datastore/manager.py +++ b/trove/guestagent/datastore/manager.py @@ -25,7 +25,6 @@ from oslo_service import periodic_task from trove.common import cfg from trove.common import exception -from trove.common import instance from trove.common.i18n import _ from trove.common.notification import EndNotification from trove.guestagent import dbaas @@ -37,6 +36,7 @@ from trove.guestagent.common.operating_system import FileMode from trove.guestagent.module import driver_manager from trove.guestagent.module import module_manager from trove.guestagent.strategies import replication as repl_strategy +from trove.instance import service_status LOG = logging.getLogger(__name__) CONF = cfg.CONF @@ -306,6 +306,10 @@ class Manager(periodic_task.PeriodicTasks): """ return {} + def upgrade(self, context, upgrade_info): + """Upgrade the database.""" + pass + def post_upgrade(self, context, upgrade_info): """Recovers the guest after the image is upgraded using information from the pre_upgrade step @@ -588,7 +592,8 @@ class Manager(periodic_task.PeriodicTasks): self.configuration_manager.apply_system_override( config_man_values, change_id=apply_label, pre_user=True) if restart_required: - self.status.set_status(instance.ServiceStatuses.RESTART_REQUIRED) + self.status.set_status( + service_status.ServiceStatuses.RESTART_REQUIRED) else: self.apply_overrides(context, cfg_values) diff --git a/trove/guestagent/datastore/mysql_common/manager.py b/trove/guestagent/datastore/mysql_common/manager.py index 2d31e2f6e3..eeb1707a16 100644 --- a/trove/guestagent/datastore/mysql_common/manager.py +++ b/trove/guestagent/datastore/mysql_common/manager.py @@ -22,7 +22,6 @@ from oslo_log import log as logging from trove.common import cfg from trove.common import configurations from trove.common import exception -from trove.common import instance as rd_instance from trove.common import utils from trove.common.notification import EndNotification from trove.guestagent import guest_log @@ -32,6 +31,7 @@ from trove.guestagent.datastore import manager from trove.guestagent.strategies import replication as repl_strategy from trove.guestagent.utils import docker as docker_util from trove.guestagent.utils import mysql as mysql_util +from trove.instance import service_status LOG = logging.getLogger(__name__) CONF = cfg.CONF @@ -71,7 +71,7 @@ class MySqlManager(manager.Manager): client.execute(cmd) LOG.debug("Database service check: database query is responsive") - return rd_instance.ServiceStatuses.HEALTHY + return service_status.ServiceStatuses.HEALTHY except Exception: return super(MySqlManager, self).get_service_status() @@ -295,7 +295,7 @@ class MySqlManager(manager.Manager): self.app.restore_backup(context, backup_info, restore_location) except Exception: LOG.error("Failed to restore from backup %s.", backup_info['id']) - self.status.set_status(rd_instance.ServiceStatuses.FAILED) + self.status.set_status(service_status.ServiceStatuses.FAILED) raise LOG.info("Finished restore data from backup %s", backup_info['id']) @@ -365,7 +365,7 @@ class MySqlManager(manager.Manager): slave_config) except Exception as err: LOG.error("Error enabling replication, error: %s", str(err)) - self.status.set_status(rd_instance.ServiceStatuses.FAILED) + self.status.set_status(service_status.ServiceStatuses.FAILED) raise def detach_replica(self, context, for_failover=False): @@ -431,3 +431,9 @@ class MySqlManager(manager.Manager): def demote_replication_master(self, context): LOG.info("Demoting replication master.") self.replication.demote_master(self.app) + + def upgrade(self, context, upgrade_info): + """Upgrade the database.""" + LOG.info('Starting to upgrade database, upgrade_info: %s', + upgrade_info) + self.app.upgrade(upgrade_info) diff --git a/trove/guestagent/datastore/mysql_common/service.py b/trove/guestagent/datastore/mysql_common/service.py index c98b327912..77b10d81e3 100644 --- a/trove/guestagent/datastore/mysql_common/service.py +++ b/trove/guestagent/datastore/mysql_common/service.py @@ -27,7 +27,6 @@ from sqlalchemy.sql.expression import text from trove.backup.state import BackupState from trove.common import cfg from trove.common import exception -from trove.common import instance from trove.common import utils from trove.common.configurations import MySQLConfParser from trove.common.db.mysql import models @@ -43,6 +42,7 @@ from trove.guestagent.datastore import service from trove.guestagent.datastore.mysql_common import service as commmon_service from trove.guestagent.utils import docker as docker_util from trove.guestagent.utils import mysql as mysql_util +from trove.instance import service_status LOG = logging.getLogger(__name__) CONF = cfg.CONF @@ -77,24 +77,24 @@ class BaseMySqlAppStatus(service.BaseDbStatus): cmd = 'mysql -uroot -p%s -e "select 1;"' % root_pass try: docker_util.run_command(self.docker_client, cmd) - return instance.ServiceStatuses.HEALTHY + return service_status.ServiceStatuses.HEALTHY except Exception as exc: LOG.warning('Failed to run docker command, error: %s', str(exc)) container_log = docker_util.get_container_logs( self.docker_client, tail='all') - LOG.warning('container log: %s', '\n'.join(container_log)) - return instance.ServiceStatuses.RUNNING + LOG.debug('container log: \n%s', '\n'.join(container_log)) + return service_status.ServiceStatuses.RUNNING elif status == "not running": - return instance.ServiceStatuses.SHUTDOWN + return service_status.ServiceStatuses.SHUTDOWN elif status == "paused": - return instance.ServiceStatuses.PAUSED + return service_status.ServiceStatuses.PAUSED elif status == "exited": - return instance.ServiceStatuses.SHUTDOWN + return service_status.ServiceStatuses.SHUTDOWN elif status == "dead": - return instance.ServiceStatuses.CRASHED + return service_status.ServiceStatuses.CRASHED else: - return instance.ServiceStatuses.UNKNOWN + return service_status.ServiceStatuses.UNKNOWN @six.add_metaclass(abc.ABCMeta) @@ -638,8 +638,9 @@ class BaseMySqlApp(object): raise exception.TroveError(_("Failed to start mysql")) if not self.status.wait_for_real_status_to_change_to( - instance.ServiceStatuses.HEALTHY, - CONF.state_change_wait_time, update_db): + service_status.ServiceStatuses.HEALTHY, + CONF.state_change_wait_time, update_db + ): raise exception.TroveError(_("Failed to start mysql")) def start_db_with_conf_changes(self, config_contents): @@ -662,7 +663,7 @@ class BaseMySqlApp(object): raise exception.TroveError("Failed to stop mysql") if not self.status.wait_for_real_status_to_change_to( - instance.ServiceStatuses.SHUTDOWN, + service_status.ServiceStatuses.SHUTDOWN, CONF.state_change_wait_time, update_db): raise exception.TroveError("Failed to stop mysql") @@ -714,7 +715,7 @@ class BaseMySqlApp(object): raise exception.TroveError("Failed to restart mysql") if not self.status.wait_for_real_status_to_change_to( - instance.ServiceStatuses.HEALTHY, + service_status.ServiceStatuses.HEALTHY, CONF.state_change_wait_time, update_db=False): raise exception.TroveError("Failed to start mysql") @@ -949,6 +950,20 @@ class BaseMySqlApp(object): q = "set global read_only = %s" % read_only client.execute(text(str(q))) + def upgrade(self, upgrade_info): + """Upgrade the database.""" + new_version = upgrade_info.get('datastore_version') + + LOG.info('Stopping db container for upgrade') + self.stop_db() + + LOG.info('Deleting db container for upgrade') + docker_util.remove_container(self.docker_client) + + LOG.info('Starting new db container with version %s for upgrade', + new_version) + self.start_db(update_db=True, ds_version=new_version) + class BaseMySqlRootAccess(object): def __init__(self, mysql_app): diff --git a/trove/guestagent/datastore/service.py b/trove/guestagent/datastore/service.py index 63a5464e44..7ac5fb0311 100644 --- a/trove/guestagent/datastore/service.py +++ b/trove/guestagent/datastore/service.py @@ -20,11 +20,11 @@ from oslo_utils import timeutils from trove.common import cfg from trove.common import context as trove_context -from trove.common import instance from trove.common.i18n import _ from trove.conductor import api as conductor_api from trove.guestagent.common import guestagent_utils from trove.guestagent.common import operating_system +from trove.instance import service_status LOG = logging.getLogger(__name__) CONF = cfg.CONF @@ -74,7 +74,7 @@ class BaseDbStatus(object): operating_system.write_file(prepare_start_file, '') self.__refresh_prepare_completed() - self.set_status(instance.ServiceStatuses.BUILDING, True) + self.set_status(service_status.ServiceStatuses.BUILDING, True) def set_ready(self): prepare_end_file = guestagent_utils.build_file_path( @@ -92,9 +92,9 @@ class BaseDbStatus(object): final_status = None if error_occurred: - final_status = instance.ServiceStatuses.FAILED + final_status = service_status.ServiceStatuses.FAILED elif post_processing: - final_status = instance.ServiceStatuses.INSTANCE_READY + final_status = service_status.ServiceStatuses.INSTANCE_READY if final_status: LOG.info("Set final status to %s.", final_status) @@ -126,8 +126,8 @@ class BaseDbStatus(object): def is_running(self): """True if DB server is running.""" return (self.status is not None and - self.status in [instance.ServiceStatuses.RUNNING, - instance.ServiceStatuses.HEALTHY]) + self.status in [service_status.ServiceStatuses.RUNNING, + service_status.ServiceStatuses.HEALTHY]) def set_status(self, status, force=False): """Use conductor to update the DB app status.""" @@ -199,7 +199,7 @@ class BaseDbStatus(object): """ LOG.debug("Waiting for database to start up.") if not self._wait_for_database_service_status( - instance.ServiceStatuses.RUNNING, timeout, update_db): + service_status.ServiceStatuses.RUNNING, timeout, update_db): raise RuntimeError(_("Database failed to start.")) LOG.info("Database has started successfully.") @@ -229,7 +229,7 @@ class BaseDbStatus(object): LOG.debug("Waiting for database to shutdown.") if not self._wait_for_database_service_status( - instance.ServiceStatuses.SHUTDOWN, timeout, update_db): + service_status.ServiceStatuses.SHUTDOWN, timeout, update_db): raise RuntimeError(_("Database failed to stop.")) LOG.info("Database has stopped successfully.") @@ -283,9 +283,19 @@ class BaseDbStatus(object): # outside. loop = True + # We need 3 (by default) consecutive success db connections for status + # 'HEALTHY' + healthy_count = 0 + while loop: self.status = self.get_actual_db_status() if self.status == status: + if (status == service_status.ServiceStatuses.HEALTHY and + healthy_count < 2): + healthy_count += 1 + time.sleep(CONF.state_change_poll_time) + continue + if update_db: self.set_status(self.status) return True diff --git a/trove/instance/models.py b/trove/instance/models.py index b5b9ac9391..7ca6a0d7fc 100644 --- a/trove/instance/models.py +++ b/trove/instance/models.py @@ -19,13 +19,13 @@ from datetime import datetime from datetime import timedelta import os.path import re -import six from novaclient import exceptions as nova_exceptions from oslo_config.cfg import NoSuchOptError from oslo_log import log as logging from oslo_utils import encodeutils from oslo_utils import netutils +import six from sqlalchemy import func from trove.backup.models import Backup @@ -33,15 +33,14 @@ from trove.common import cfg from trove.common import clients from trove.common import crypto_utils as cu from trove.common import exception -from trove.common.i18n import _ -from trove.common import instance as tr_instance from trove.common import neutron from trove.common import notification from trove.common import server_group as srv_grp from trove.common import template from trove.common import timeutils -from trove.common.trove_remote import create_trove_client from trove.common import utils +from trove.common.i18n import _ +from trove.common.trove_remote import create_trove_client from trove.configuration.models import Configuration from trove.datastore import models as datastore_models from trove.datastore.models import DatastoreVersionMetadata as dvm @@ -49,6 +48,7 @@ from trove.datastore.models import DBDatastoreVersionMetadata from trove.db import get_db_api from trove.db import models as dbmodels from trove.extensions.security_group.models import SecurityGroup +from trove.instance import service_status as srvstatus from trove.instance.tasks import InstanceTask from trove.instance.tasks import InstanceTasks from trove.module import models as module_models @@ -339,25 +339,25 @@ class SimpleInstance(object): action = self.db_info.task_status.action # Check if we are resetting status or force deleting - if (tr_instance.ServiceStatuses.UNKNOWN == self.datastore_status.status + if (srvstatus.ServiceStatuses.UNKNOWN == self.datastore_status.status and action == InstanceTasks.DELETING.action): return InstanceStatus.SHUTDOWN - elif (tr_instance.ServiceStatuses.UNKNOWN == + elif (srvstatus.ServiceStatuses.UNKNOWN == self.datastore_status.status): return InstanceStatus.ERROR # Check for taskmanager status. - if 'BUILDING' == action: + if InstanceTasks.BUILDING.action == action: if 'ERROR' == self.db_info.server_status: return InstanceStatus.ERROR return InstanceStatus.BUILD - if 'REBOOTING' == action: + if InstanceTasks.REBOOTING.action == action: return InstanceStatus.REBOOT - if 'RESIZING' == action: + if InstanceTasks.RESIZING.action == action: return InstanceStatus.RESIZE - if 'UPGRADING' == action: + if InstanceTasks.UPGRADING.action == action: return InstanceStatus.UPGRADE - if 'RESTART_REQUIRED' == action: + if InstanceTasks.RESTART_REQUIRED.action == action: return InstanceStatus.RESTART_REQUIRED if InstanceTasks.PROMOTING.action == action: return InstanceStatus.PROMOTE @@ -396,10 +396,10 @@ class SimpleInstance(object): # Check against the service status. # The service is only paused during a reboot. - if tr_instance.ServiceStatuses.PAUSED == self.datastore_status.status: + if srvstatus.ServiceStatuses.PAUSED == self.datastore_status.status: return InstanceStatus.REBOOT # If the service status is NEW, then we are building. - if tr_instance.ServiceStatuses.NEW == self.datastore_status.status: + if srvstatus.ServiceStatuses.NEW == self.datastore_status.status: return InstanceStatus.BUILD # For everything else we can look at the service status mapping. @@ -594,14 +594,19 @@ def load_instance(cls, context, id, needs_server=False, def load_instance_with_info(cls, context, id, cluster_id=None): db_info = get_db_info(context, id, cluster_id) + LOG.debug('Task status for instance %s: %s', id, db_info.task_status) + + service_status = InstanceServiceStatus.find_by(instance_id=id) + if (db_info.task_status == InstanceTasks.NONE and + not service_status.is_uptodate()): + LOG.warning('Guest agent heartbeat for instance %s has expried', id) + service_status.status = \ + srvstatus.ServiceStatuses.FAILED_TIMEOUT_GUESTAGENT load_simple_instance_server_status(context, db_info) load_simple_instance_addresses(context, db_info) - service_status = InstanceServiceStatus.find_by(instance_id=id) - LOG.debug("Instance %(instance_id)s service status is %(service_status)s.", - {'instance_id': id, 'service_status': service_status.status}) instance = cls(context, db_info, service_status) load_guest_info(instance, context, id) @@ -879,7 +884,7 @@ class BaseInstance(SimpleInstance): def set_servicestatus_deleted(self): del_instance = InstanceServiceStatus.find_by(instance_id=self.id) - del_instance.set_status(tr_instance.ServiceStatuses.DELETED) + del_instance.set_status(srvstatus.ServiceStatuses.DELETED) del_instance.save() def set_instance_fault_deleted(self): @@ -956,7 +961,12 @@ class BaseInstance(SimpleInstance): self.reset_task_status() reset_instance = InstanceServiceStatus.find_by(instance_id=self.id) - reset_instance.set_status(tr_instance.ServiceStatuses.UNKNOWN) + reset_instance.set_status(srvstatus.ServiceStatuses.UNKNOWN) + reset_instance.save() + + def set_service_status(self, status): + reset_instance = InstanceServiceStatus.find_by(instance_id=self.id) + reset_instance.set_status(status) reset_instance.save() @@ -1267,7 +1277,7 @@ class Instance(BuiltInstance): overrides = config.get_configuration_overrides() service_status = InstanceServiceStatus.create( instance_id=instance_id, - status=tr_instance.ServiceStatuses.NEW) + status=srvstatus.ServiceStatuses.NEW) if CONF.trove_dns_support: dns_client = clients.create_dns_client(context) @@ -1762,19 +1772,32 @@ class Instances(object): db.server_status = "SHUTDOWN" # Fake it... db.addresses = [] - # volumes = find_volumes(server.id) datastore_status = InstanceServiceStatus.find_by( instance_id=db.id) if not datastore_status.status: # This should never happen. LOG.error("Server status could not be read for " "instance id(%s).", db.id) continue - LOG.debug("Server api_status(%s).", - datastore_status.status.api_status) + + # Get the real-time service status. + LOG.debug('Task status for instance %s: %s', db.id, + db.task_status) + if db.task_status == InstanceTasks.NONE: + last_heartbeat_delta = ( + timeutils.utcnow() - datastore_status.updated_at) + agent_expiry_interval = timedelta( + seconds=CONF.agent_heartbeat_expiry) + if last_heartbeat_delta > agent_expiry_interval: + LOG.warning( + 'Guest agent heartbeat for instance %s has ' + 'expried', id) + datastore_status.status = \ + srvstatus.ServiceStatuses.FAILED_TIMEOUT_GUESTAGENT except exception.ModelNotFoundError: LOG.error("Server status could not be read for " "instance id(%s).", db.id) continue + ret.append(load_instance(context, db, datastore_status, server=server)) return ret @@ -2001,7 +2024,7 @@ class InstanceServiceStatus(dbmodels.DatabaseModelBase): def _validate(self, errors): if self.status is None: errors['status'] = "Cannot be None." - if tr_instance.ServiceStatus.from_code(self.status_id) is None: + if srvstatus.ServiceStatus.from_code(self.status_id) is None: errors['status_id'] = "Not valid." def get_status(self): @@ -2012,7 +2035,7 @@ class InstanceServiceStatus(dbmodels.DatabaseModelBase): status of the service :rtype: trove.common.instance.ServiceStatus """ - return tr_instance.ServiceStatus.from_code(self.status_id) + return srvstatus.ServiceStatus.from_code(self.status_id) def set_status(self, value): """ @@ -2027,6 +2050,15 @@ class InstanceServiceStatus(dbmodels.DatabaseModelBase): self['updated_at'] = timeutils.utcnow() return get_db_api().save(self) + def is_uptodate(self): + """Check if the service status heartbeat is up to date.""" + heartbeat_expiry = timedelta(seconds=CONF.agent_heartbeat_expiry) + last_update = (timeutils.utcnow() - self.updated_at) + if last_update < heartbeat_expiry: + return True + + return False + status = property(get_status, set_status) @@ -2039,6 +2071,6 @@ def persisted_models(): MYSQL_RESPONSIVE_STATUSES = [ - tr_instance.ServiceStatuses.RUNNING, - tr_instance.ServiceStatuses.HEALTHY + srvstatus.ServiceStatuses.RUNNING, + srvstatus.ServiceStatuses.HEALTHY ] diff --git a/trove/common/instance.py b/trove/instance/service_status.py similarity index 98% rename from trove/common/instance.py rename to trove/instance/service_status.py index eb7cf7e02e..001c565dcd 100644 --- a/trove/common/instance.py +++ b/trove/instance/service_status.py @@ -104,6 +104,7 @@ class ServiceStatuses(object): RESTART_REQUIRED = ServiceStatus(0x20, 'restart required', 'RESTART_REQUIRED') HEALTHY = ServiceStatus(0x21, 'healthy', 'HEALTHY') + UPGRADING = ServiceStatus(0x22, 'upgrading', 'UPGRADING') # Dissuade further additions at run-time. diff --git a/trove/taskmanager/models.py b/trove/taskmanager/models.py index 0e0abe9956..104e15fb0b 100755 --- a/trove/taskmanager/models.py +++ b/trove/taskmanager/models.py @@ -14,6 +14,7 @@ import copy import os.path +import time import traceback from cinderclient import exceptions as cinder_exceptions @@ -21,7 +22,6 @@ from eventlet import greenthread from eventlet.timeout import Timeout from oslo_log import log as logging from swiftclient.client import ClientException -import time from trove import rpc from trove.backup import models as bkup_models @@ -33,9 +33,7 @@ from trove.cluster.models import Cluster from trove.cluster.models import DBCluster from trove.common import cfg from trove.common import clients -from trove.common import crypto_utils as cu from trove.common import exception -from trove.common import instance as rd_instance from trove.common import neutron from trove.common import template from trove.common import timeutils @@ -51,7 +49,6 @@ from trove.common.exception import PollTimeOut from trove.common.exception import TroveError from trove.common.exception import VolumeCreationFailure from trove.common.i18n import _ -from trove.common.instance import ServiceStatuses from trove.common.notification import DBaaSInstanceRestart from trove.common.notification import DBaaSInstanceUpgrade from trove.common.notification import EndNotification @@ -63,6 +60,7 @@ from trove.common.strategies.cluster import strategy from trove.common.utils import try_recover from trove.extensions.mysql import models as mysql_models from trove.instance import models as inst_models +from trove.instance import service_status as srvstatus from trove.instance.models import BuiltInstance from trove.instance.models import DBInstance from trove.instance.models import FreshInstance @@ -202,24 +200,36 @@ class ClusterTasks(Cluster): shard_id=None): """Wait for all instances to get READY.""" return self._all_instances_acquire_status( - instance_ids, cluster_id, shard_id, ServiceStatuses.INSTANCE_READY, - fast_fail_statuses=[ServiceStatuses.FAILED, - ServiceStatuses.FAILED_TIMEOUT_GUESTAGENT]) + instance_ids, cluster_id, shard_id, + srvstatus.ServiceStatuses.INSTANCE_READY, + fast_fail_statuses=[ + srvstatus.ServiceStatuses.FAILED, + srvstatus.ServiceStatuses.FAILED_TIMEOUT_GUESTAGENT + ] + ) def _all_instances_shutdown(self, instance_ids, cluster_id, shard_id=None): """Wait for all instances to go SHUTDOWN.""" return self._all_instances_acquire_status( - instance_ids, cluster_id, shard_id, ServiceStatuses.SHUTDOWN, - fast_fail_statuses=[ServiceStatuses.FAILED, - ServiceStatuses.FAILED_TIMEOUT_GUESTAGENT]) + instance_ids, cluster_id, shard_id, + srvstatus.ServiceStatuses.SHUTDOWN, + fast_fail_statuses=[ + srvstatus.ServiceStatuses.FAILED, + srvstatus.ServiceStatuses.FAILED_TIMEOUT_GUESTAGENT + ] + ) def _all_instances_running(self, instance_ids, cluster_id, shard_id=None): """Wait for all instances to become ACTIVE.""" return self._all_instances_acquire_status( - instance_ids, cluster_id, shard_id, ServiceStatuses.RUNNING, - fast_fail_statuses=[ServiceStatuses.FAILED, - ServiceStatuses.FAILED_TIMEOUT_GUESTAGENT]) + instance_ids, cluster_id, shard_id, + srvstatus.ServiceStatuses.RUNNING, + fast_fail_statuses=[ + srvstatus.ServiceStatuses.FAILED, + srvstatus.ServiceStatuses.FAILED_TIMEOUT_GUESTAGENT + ] + ) def _all_instances_acquire_status( self, instance_ids, cluster_id, shard_id, expected_status, @@ -427,7 +437,10 @@ class FreshInstanceTasks(FreshInstance, NotifyMixin, ConfigurationMixin): utils.poll_until(self._service_is_active, sleep_time=CONF.usage_sleep_time, time_out=timeout) + LOG.info("Created instance %s successfully.", self.id) + if not self.db_info.task_status.is_error: + self.reset_task_status() TroveInstanceCreate(instance=self, instance_size=flavor['ram']).notify() except (TroveError, PollTimeOut) as ex: @@ -582,9 +595,6 @@ class FreshInstanceTasks(FreshInstance, NotifyMixin, ConfigurationMixin): if root_password: self.report_root_enabled() - if not self.db_info.task_status.is_error: - self.reset_task_status() - # when DNS is supported, we attempt to add this after the # instance is prepared. Otherwise, if DNS fails, instances # end up in a poorer state and there's no tooling around @@ -723,12 +733,13 @@ class FreshInstanceTasks(FreshInstance, NotifyMixin, ConfigurationMixin): if CONF.update_status_on_fail: # Updating service status service = InstanceServiceStatus.find_by(instance_id=self.id) - service.set_status(ServiceStatuses.FAILED_TIMEOUT_GUESTAGENT) + service.set_status( + srvstatus.ServiceStatuses.FAILED_TIMEOUT_GUESTAGENT) service.save() LOG.error( "Service status: %s, service error description: %s", - ServiceStatuses.FAILED_TIMEOUT_GUESTAGENT.api_status, - ServiceStatuses.FAILED_TIMEOUT_GUESTAGENT.description + srvstatus.ServiceStatuses.FAILED_TIMEOUT_GUESTAGENT.api_status, + srvstatus.ServiceStatuses.FAILED_TIMEOUT_GUESTAGENT.description ) # Updating instance status @@ -756,14 +767,14 @@ class FreshInstanceTasks(FreshInstance, NotifyMixin, ConfigurationMixin): service = InstanceServiceStatus.find_by(instance_id=self.id) status = service.get_status() - if (status == rd_instance.ServiceStatuses.RUNNING or - status == rd_instance.ServiceStatuses.INSTANCE_READY or - status == rd_instance.ServiceStatuses.HEALTHY): + if (status == srvstatus.ServiceStatuses.RUNNING or + status == srvstatus.ServiceStatuses.INSTANCE_READY or + status == srvstatus.ServiceStatuses.HEALTHY): return True - elif status not in [rd_instance.ServiceStatuses.NEW, - rd_instance.ServiceStatuses.BUILDING, - rd_instance.ServiceStatuses.UNKNOWN, - rd_instance.ServiceStatuses.DELETED]: + elif status not in [srvstatus.ServiceStatuses.NEW, + srvstatus.ServiceStatuses.BUILDING, + srvstatus.ServiceStatuses.UNKNOWN, + srvstatus.ServiceStatuses.DELETED]: raise TroveError(_("Service not active, status: %s") % status) c_id = self.db_info.compute_instance_id @@ -1069,6 +1080,27 @@ class BuiltInstanceTasks(BuiltInstance, NotifyMixin, ConfigurationMixin): associated with a compute server. """ + def is_service_healthy(self): + """Wait for the db service up and running. + + This method is supposed to be called with poll_until against an + existing db instance. + """ + service = InstanceServiceStatus.find_by(instance_id=self.id) + status = service.get_status() + + if service.is_uptodate(): + if status in [srvstatus.ServiceStatuses.HEALTHY]: + return True + elif status in [ + srvstatus.ServiceStatuses.FAILED, + srvstatus.ServiceStatuses.UNKNOWN, + srvstatus.ServiceStatuses.FAILED_TIMEOUT_GUESTAGENT + ]: + raise TroveError('Database service error, status: %s' % status) + + return False + def resize_volume(self, new_size): LOG.info("Resizing volume for instance %(instance_id)s from " "%(old_size)s GB to %(new_size)s GB.", @@ -1219,6 +1251,10 @@ class BuiltInstanceTasks(BuiltInstance, NotifyMixin, ConfigurationMixin): LOG.info("Starting database on instance %s.", self.id) self.guest.restart() + # Wait for database service up and running + utils.poll_until(self.is_service_healthy, + time_out=CONF.report_interval * 2) + LOG.info("Rebooted instance %s successfully.", self.id) except Exception as e: LOG.error("Failed to reboot instance %(id)s: %(e)s", @@ -1276,79 +1312,40 @@ class BuiltInstanceTasks(BuiltInstance, NotifyMixin, ConfigurationMixin): This does not change the reference for this BuiltInstanceTask """ datastore_status = InstanceServiceStatus.find_by(instance_id=self.id) - datastore_status.status = rd_instance.ServiceStatuses.PAUSED + datastore_status.status = srvstatus.ServiceStatuses.PAUSED datastore_status.save() def upgrade(self, datastore_version): LOG.info("Upgrading instance %s to new datastore version %s", self.id, datastore_version) - def server_finished_rebuilding(): - self.refresh_compute_server_info() - return not self.server_status_matches(['REBUILD']) + self.set_service_status(srvstatus.ServiceStatuses.UPGRADING) try: upgrade_info = self.guest.pre_upgrade() + upgrade_info = upgrade_info if upgrade_info else {} + upgrade_info.update({'datastore_version': datastore_version.name}) + self.guest.upgrade(upgrade_info) - if self.volume_id: - volume = self.volume_client.volumes.get(self.volume_id) - volume_device = self._fix_device_path( - volume.attachments[0]['device']) - if volume: - upgrade_info['device'] = volume_device - - # BUG(1650518): Cleanup in the Pike release some instances - # that we will be upgrading will be pre secureserialier - # and will have no instance_key entries. If this is one of - # those instances, make a key. That will make it appear in - # the injected files that are generated next. From this - # point, and until the guest comes up, attempting to send - # messages to it will fail because the RPC framework will - # encrypt messages to a guest which potentially doesn't - # have the code to handle it. - if CONF.enable_secure_rpc_messaging and ( - self.db_info.encrypted_key is None): - encrypted_key = cu.encode_data(cu.encrypt_data( - cu.generate_random_key(), - CONF.inst_rpc_key_encr_key)) - self.update_db(encrypted_key=encrypted_key) - LOG.debug("Generated unique RPC encryption key for " - "instance = %(id)s, key = %(key)s", - {'id': self.id, 'key': encrypted_key}) - - injected_files = self.get_injected_files( - datastore_version.manager) - LOG.debug("Rebuilding instance %(instance)s with image %(image)s.", - {'instance': self, 'image': datastore_version.image_id}) - self.server.rebuild(datastore_version.image_id, - files=injected_files) - utils.poll_until( - server_finished_rebuilding, - sleep_time=5, time_out=600) - - if not self.server_status_matches(['ACTIVE']): - raise TroveError(_("Instance %(instance)s failed to " - "upgrade to %(datastore_version)s"), - instance=self, - datastore_version=datastore_version) - - LOG.info('Finished rebuilding server for instance %s', self.id) - - self.guest.post_upgrade(upgrade_info) + # Wait for db instance healthy + LOG.info('Waiting for instance %s to be healthy after upgrading', + self.id) + utils.poll_until(self.is_service_healthy, time_out=600, + sleep_time=5) self.reset_task_status() LOG.info("Finished upgrading instance %s to new datastore " - "version %s", - self.id, datastore_version) + "version %s", self.id, datastore_version) except Exception as e: - LOG.exception(e) - err = inst_models.InstanceTasks.BUILDING_ERROR_SERVER - self.update_db(task_status=err) - raise e + LOG.error('Failed to upgrade instance %s, error: %s', self.id, e) + self.update_db( + task_status=inst_models.InstanceTasks.BUILDING_ERROR_SERVER) - # Some cinder drivers appear to return "vdb" instead of "/dev/vdb". - # We need to account for that. def _fix_device_path(self, device): + """Get correct device path. + + Some cinder drivers appear to return "vdb" instead of "/dev/vdb". + """ if device.startswith("/dev"): return device else: @@ -1515,7 +1512,7 @@ class ResizeVolumeAction(object): "status to failed.", {'func': orig_func.__name__, 'id': self.instance.id}) service = InstanceServiceStatus.find_by(instance_id=self.instance.id) - service.set_status(ServiceStatuses.FAILED) + service.set_status(srvstatus.ServiceStatuses.FAILED) service.save() def _recover_restart(self, orig_func): @@ -1790,7 +1787,7 @@ class ResizeActionBase(object): def _datastore_is_offline(self): self.instance._refresh_datastore_status() return (self.instance.datastore_status_matches( - rd_instance.ServiceStatuses.SHUTDOWN)) + srvstatus.ServiceStatuses.SHUTDOWN)) def _revert_nova_action(self): LOG.debug("Instance %s calling Compute revert resize...", @@ -1811,7 +1808,7 @@ class ResizeActionBase(object): def _guest_is_awake(self): self.instance._refresh_datastore_status() return not self.instance.datastore_status_matches( - rd_instance.ServiceStatuses.PAUSED) + srvstatus.ServiceStatuses.PAUSED) def _perform_nova_action(self): """Calls Nova to resize or migrate an instance, and confirms.""" diff --git a/trove/tests/api/instances_actions.py b/trove/tests/api/instances_actions.py index 0c0f068002..f8d9f0affb 100644 --- a/trove/tests/api/instances_actions.py +++ b/trove/tests/api/instances_actions.py @@ -267,6 +267,15 @@ class RebootTestBase(ActionTestBase): poll_until(is_finished_rebooting, time_out=TIME_OUT_TIME) + def wait_for_status(self, status, timeout=60): + def is_status(): + instance = self.instance + if instance.status in status: + return True + return False + + poll_until(is_status, time_out=timeout) + @test(groups=[tests.DBAAS_API_INSTANCE_ACTIONS], depends_on_groups=[tests.DBAAS_API_DATABASES], @@ -312,9 +321,9 @@ class StopTests(RebootTestBase): @test(depends_on=[test_ensure_mysql_is_running]) def test_stop_mysql(self): - """Stops MySQL.""" + """Stops MySQL by admin.""" instance_info.dbaas_admin.management.stop(self.instance_id) - self.wait_for_failure_status() + self.wait_for_status(['SHUTDOWN'], timeout=60) @test(depends_on=[test_stop_mysql]) def test_volume_info_while_mysql_is_down(self): diff --git a/trove/tests/api/instances_resize.py b/trove/tests/api/instances_resize.py index 0d00a9cb52..f9bec976fe 100644 --- a/trove/tests/api/instances_resize.py +++ b/trove/tests/api/instances_resize.py @@ -13,21 +13,21 @@ # License for the specific language governing permissions and limitations # under the License. -from novaclient.exceptions import BadRequest -from novaclient.v2.servers import Server from unittest import mock +from novaclient.exceptions import BadRequest +from novaclient.v2.servers import Server from oslo_messaging._drivers.common import RPCException from proboscis import test from testtools import TestCase -from trove.common.exception import PollTimeOut -from trove.common.exception import TroveError -from trove.common import instance as rd_instance from trove.common import template from trove.common import utils +from trove.common.exception import PollTimeOut +from trove.common.exception import TroveError from trove.datastore.models import DatastoreVersion from trove.guestagent import api as guest +from trove.instance import service_status as srvstatus from trove.instance.models import DBInstance from trove.instance.models import InstanceServiceStatus from trove.instance.tasks import InstanceTasks @@ -63,7 +63,7 @@ class ResizeTestBase(TestCase): self.server, datastore_status=InstanceServiceStatus.create( instance_id=self.db_info.id, - status=rd_instance.ServiceStatuses.RUNNING)) + status=srvstatus.ServiceStatuses.RUNNING)) self.instance.server.flavor = {'id': OLD_FLAVOR_ID} self.guest = mock.MagicMock(spec=guest.API) self.instance._guest = self.guest @@ -124,7 +124,7 @@ class ResizeTests(ResizeTestBase): task_status=InstanceTasks.NONE) def test_nova_wont_resize(self): - self._datastore_changes_to(rd_instance.ServiceStatuses.SHUTDOWN) + self._datastore_changes_to(srvstatus.ServiceStatuses.SHUTDOWN) self.server.resize.side_effect = BadRequest(400) self.server.status = "ACTIVE" self.assertRaises(BadRequest, self.action.execute) @@ -135,7 +135,7 @@ class ResizeTests(ResizeTestBase): task_status=InstanceTasks.NONE) def test_nova_resize_timeout(self): - self._datastore_changes_to(rd_instance.ServiceStatuses.SHUTDOWN) + self._datastore_changes_to(srvstatus.ServiceStatuses.SHUTDOWN) self.server.status = "ACTIVE" with mock.patch.object(utils, 'poll_until') as mock_poll_until: @@ -150,7 +150,7 @@ class ResizeTests(ResizeTestBase): task_status=InstanceTasks.NONE) def test_nova_doesnt_change_flavor(self): - self._datastore_changes_to(rd_instance.ServiceStatuses.SHUTDOWN) + self._datastore_changes_to(srvstatus.ServiceStatuses.SHUTDOWN) with mock.patch.object(utils, 'poll_until') as mock_poll_until: self.poll_until_side_effects.extend([ @@ -177,7 +177,7 @@ class ResizeTests(ResizeTestBase): task_status=InstanceTasks.NONE) def test_nova_resize_fails(self): - self._datastore_changes_to(rd_instance.ServiceStatuses.SHUTDOWN) + self._datastore_changes_to(srvstatus.ServiceStatuses.SHUTDOWN) with mock.patch.object(utils, 'poll_until') as mock_poll_until: self.poll_until_side_effects.extend([ @@ -200,7 +200,7 @@ class ResizeTests(ResizeTestBase): task_status=InstanceTasks.NONE) def test_nova_resizes_in_weird_state(self): - self._datastore_changes_to(rd_instance.ServiceStatuses.SHUTDOWN) + self._datastore_changes_to(srvstatus.ServiceStatuses.SHUTDOWN) with mock.patch.object(utils, 'poll_until') as mock_poll_until: self.poll_until_side_effects.extend([ @@ -224,7 +224,7 @@ class ResizeTests(ResizeTestBase): task_status=InstanceTasks.NONE) def test_guest_is_not_okay(self): - self._datastore_changes_to(rd_instance.ServiceStatuses.SHUTDOWN) + self._datastore_changes_to(srvstatus.ServiceStatuses.SHUTDOWN) with mock.patch.object(utils, 'poll_until') as mock_poll_until: self.poll_until_side_effects.extend([ @@ -237,7 +237,7 @@ class ResizeTests(ResizeTestBase): self.instance.set_datastore_status_to_paused.side_effect = ( lambda: self._datastore_changes_to( - rd_instance.ServiceStatuses.PAUSED)) + srvstatus.ServiceStatuses.PAUSED)) self.assertRaises(PollTimeOut, self.action.execute) @@ -257,7 +257,7 @@ class ResizeTests(ResizeTestBase): task_status=InstanceTasks.NONE) def test_mysql_is_not_okay(self): - self._datastore_changes_to(rd_instance.ServiceStatuses.SHUTDOWN) + self._datastore_changes_to(srvstatus.ServiceStatuses.SHUTDOWN) with mock.patch.object(utils, 'poll_until') as mock_poll_until: self.poll_until_side_effects.extend([ @@ -269,7 +269,7 @@ class ResizeTests(ResizeTestBase): self.instance.set_datastore_status_to_paused.side_effect = ( lambda: self._datastore_changes_to( - rd_instance.ServiceStatuses.SHUTDOWN)) + srvstatus.ServiceStatuses.SHUTDOWN)) self._start_mysql() self.assertRaises(PollTimeOut, self.action.execute) @@ -290,7 +290,7 @@ class ResizeTests(ResizeTestBase): task_status=InstanceTasks.NONE) def test_confirm_resize_fails(self): - self._datastore_changes_to(rd_instance.ServiceStatuses.SHUTDOWN) + self._datastore_changes_to(srvstatus.ServiceStatuses.SHUTDOWN) with mock.patch.object(utils, 'poll_until') as mock_poll_until: self.poll_until_side_effects.extend([ @@ -303,7 +303,7 @@ class ResizeTests(ResizeTestBase): self.instance.set_datastore_status_to_paused.side_effect = ( lambda: self._datastore_changes_to( - rd_instance.ServiceStatuses.RUNNING)) + srvstatus.ServiceStatuses.RUNNING)) self.server.confirm_resize.side_effect = BadRequest(400) self._start_mysql() @@ -322,7 +322,7 @@ class ResizeTests(ResizeTestBase): task_status=InstanceTasks.NONE) def test_revert_nova_fails(self): - self._datastore_changes_to(rd_instance.ServiceStatuses.SHUTDOWN) + self._datastore_changes_to(srvstatus.ServiceStatuses.SHUTDOWN) with mock.patch.object(utils, 'poll_until') as mock_poll_until: self.poll_until_side_effects.extend([ @@ -335,7 +335,7 @@ class ResizeTests(ResizeTestBase): self.instance.set_datastore_status_to_paused.side_effect = ( lambda: self._datastore_changes_to( - rd_instance.ServiceStatuses.PAUSED)) + srvstatus.ServiceStatuses.PAUSED)) self.assertRaises(PollTimeOut, self.action.execute) @@ -363,7 +363,7 @@ class MigrateTests(ResizeTestBase): self.action = models.MigrateAction(self.instance) def test_successful_migrate(self): - self._datastore_changes_to(rd_instance.ServiceStatuses.SHUTDOWN) + self._datastore_changes_to(srvstatus.ServiceStatuses.SHUTDOWN) with mock.patch.object(utils, 'poll_until') as mock_poll_until: self.poll_until_side_effects.extend([ @@ -375,7 +375,7 @@ class MigrateTests(ResizeTestBase): self.instance.set_datastore_status_to_paused.side_effect = ( lambda: self._datastore_changes_to( - rd_instance.ServiceStatuses.RUNNING)) + srvstatus.ServiceStatuses.RUNNING)) self.action.execute() diff --git a/trove/tests/api/mgmt/instances_actions.py b/trove/tests/api/mgmt/instances_actions.py index 798d5547c9..82097a59ad 100644 --- a/trove/tests/api/mgmt/instances_actions.py +++ b/trove/tests/api/mgmt/instances_actions.py @@ -12,23 +12,24 @@ # License for the specific language governing permissions and limitations # under the License. +from unittest import mock + from novaclient.v2.servers import Server from proboscis import after_class -from proboscis.asserts import assert_equal -from proboscis.asserts import assert_raises from proboscis import before_class from proboscis import SkipTest from proboscis import test -from unittest import mock +from proboscis.asserts import assert_equal +from proboscis.asserts import assert_raises from trove.backup import models as backup_models from trove.backup import state -from trove.common.context import TroveContext from trove.common import exception -import trove.common.instance as tr_instance +from trove.common.context import TroveContext from trove.extensions.mgmt.instances.models import MgmtInstance from trove.extensions.mgmt.instances.service import MgmtInstanceController from trove.instance import models as imodels +from trove.instance import service_status as srvstatus from trove.instance.models import DBInstance from trove.instance.tasks import InstanceTasks from trove.tests.config import CONFIG @@ -65,7 +66,7 @@ class MgmtInstanceBase(object): self.db_info, self.server, datastore_status=imodels.InstanceServiceStatus( - tr_instance.ServiceStatuses.RUNNING)) + srvstatus.ServiceStatuses.RUNNING)) def _make_request(self, path='/', context=None, **kwargs): from webob import Request diff --git a/trove/tests/fakes/guestagent.py b/trove/tests/fakes/guestagent.py index 72e30cbf07..a711058069 100644 --- a/trove/tests/fakes/guestagent.py +++ b/trove/tests/fakes/guestagent.py @@ -20,7 +20,7 @@ import eventlet from oslo_log import log as logging from trove.common import exception as rd_exception -from trove.common import instance as rd_instance +from trove.instance import service_status as srvstatus from trove.tests.util import unquote_user_host DB = {} @@ -236,9 +236,9 @@ class FakeGuest(object): def update_db(): status = InstanceServiceStatus.find_by(instance_id=self.id) if instance_name.endswith('GUEST_ERROR'): - status.status = rd_instance.ServiceStatuses.FAILED + status.status = srvstatus.ServiceStatuses.FAILED else: - status.status = rd_instance.ServiceStatuses.HEALTHY + status.status = srvstatus.ServiceStatuses.HEALTHY status.save() AgentHeartBeat.create(instance_id=self.id) eventlet.spawn_after(3.5, update_db) @@ -246,8 +246,8 @@ class FakeGuest(object): def _set_task_status(self, new_status='HEALTHY'): from trove.instance.models import InstanceServiceStatus print("Setting status to %s" % new_status) - states = {'HEALTHY': rd_instance.ServiceStatuses.HEALTHY, - 'SHUTDOWN': rd_instance.ServiceStatuses.SHUTDOWN, + states = {'HEALTHY': srvstatus.ServiceStatuses.HEALTHY, + 'SHUTDOWN': srvstatus.ServiceStatuses.SHUTDOWN, } status = InstanceServiceStatus.find_by(instance_id=self.id) status.status = states[new_status] diff --git a/trove/tests/fakes/nova.py b/trove/tests/fakes/nova.py index b4aa9b434b..cc896c5c55 100644 --- a/trove/tests/fakes/nova.py +++ b/trove/tests/fakes/nova.py @@ -13,18 +13,17 @@ # License for the specific language governing permissions and limitations # under the License. +import collections +import uuid + +import eventlet from novaclient import exceptions as nova_exceptions from oslo_log import log as logging from trove.common.exception import PollTimeOut -from trove.common import instance as rd_instance +from trove.instance import service_status as srvstatus from trove.tests.fakes.common import authorize -import collections -import eventlet -import uuid - - LOG = logging.getLogger(__name__) FAKE_HOSTS = ["fake_host_1", "fake_host_2"] @@ -326,7 +325,7 @@ class FakeServers(object): instance = DBInstance.find_by(compute_instance_id=id) LOG.debug("Setting server %s to running", instance.id) status = InstanceServiceStatus.find_by(instance_id=instance.id) - status.status = rd_instance.ServiceStatuses.RUNNING + status.status = srvstatus.ServiceStatuses.RUNNING status.save() eventlet.spawn_after(time_from_now, set_server_running) diff --git a/trove/tests/unittests/conductor/test_methods.py b/trove/tests/unittests/conductor/test_methods.py index 56278adce3..64ba117ec0 100644 --- a/trove/tests/unittests/conductor/test_methods.py +++ b/trove/tests/unittests/conductor/test_methods.py @@ -18,14 +18,13 @@ from oslo_utils import timeutils from trove.backup import models as bkup_models from trove.backup import state from trove.common import exception as t_exception -from trove.common.instance import ServiceStatuses from trove.common import utils from trove.conductor import manager as conductor_manager from trove.instance import models as t_models +from trove.instance.service_status import ServiceStatuses from trove.tests.unittests import trove_testtools from trove.tests.unittests.util import util - # See LP bug #1255178 OLD_DBB_SAVE = bkup_models.DBBackup.save diff --git a/trove/tests/unittests/instance/test_instance_models.py b/trove/tests/unittests/instance/test_instance_models.py index 7c4ca2822a..eab5901bfb 100644 --- a/trove/tests/unittests/instance/test_instance_models.py +++ b/trove/tests/unittests/instance/test_instance_models.py @@ -11,15 +11,14 @@ # 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 mock import Mock +from mock import patch import uuid -from mock import Mock, patch - from trove.backup import models as backup_models from trove.common import cfg from trove.common import clients from trove.common import exception -from trove.common.instance import ServiceStatuses from trove.common import neutron from trove.datastore import models as datastore_models from trove.instance import models @@ -29,6 +28,7 @@ from trove.instance.models import Instance from trove.instance.models import instance_encryption_key_cache from trove.instance.models import InstanceServiceStatus from trove.instance.models import SimpleInstance +from trove.instance.service_status import ServiceStatuses from trove.instance.tasks import InstanceTasks from trove.taskmanager import api as task_api from trove.tests.fakes import nova diff --git a/trove/tests/unittests/instance/test_instance_status.py b/trove/tests/unittests/instance/test_instance_status.py index f52767de84..a81849ce8b 100644 --- a/trove/tests/unittests/instance/test_instance_status.py +++ b/trove/tests/unittests/instance/test_instance_status.py @@ -13,15 +13,16 @@ # License for the specific language governing permissions and limitations # under the License. # -from trove.common.instance import ServiceStatuses +import uuid + from trove.datastore import models from trove.instance.models import InstanceServiceStatus from trove.instance.models import InstanceStatus from trove.instance.models import SimpleInstance +from trove.instance.service_status import ServiceStatuses from trove.instance.tasks import InstanceTasks from trove.tests.unittests import trove_testtools from trove.tests.unittests.util import util -import uuid class FakeInstanceTask(object): diff --git a/trove/tests/unittests/mgmt/test_models.py b/trove/tests/unittests/mgmt/test_models.py index 66d698e6ea..e51e7d704b 100644 --- a/trove/tests/unittests/mgmt/test_models.py +++ b/trove/tests/unittests/mgmt/test_models.py @@ -13,26 +13,32 @@ # License for the specific language governing permissions and limitations # under the License. # +from testtools.matchers import Equals +from testtools.matchers import Is +from testtools.matchers import Not import uuid -from mock import MagicMock, patch, ANY +from mock import ANY +from mock import MagicMock +from mock import patch from novaclient.client import Client -from novaclient.v2.flavors import FlavorManager, Flavor -from novaclient.v2.servers import Server, ServerManager +from novaclient.v2.flavors import Flavor +from novaclient.v2.flavors import FlavorManager +from novaclient.v2.servers import Server +from novaclient.v2.servers import ServerManager from oslo_config import cfg -from testtools.matchers import Equals, Is, Not +from trove import rpc from trove.backup.models import Backup from trove.common import clients from trove.common import exception -from trove.common import instance as rd_instance from trove.datastore import models as datastore_models import trove.extensions.mgmt.instances.models as mgmtmodels from trove.guestagent.api import API +from trove.instance import service_status as srvstatus from trove.instance.models import DBInstance from trove.instance.models import InstanceServiceStatus from trove.instance.tasks import InstanceTasks -from trove import rpc from trove.tests.unittests import trove_testtools from trove.tests.unittests.util import util @@ -98,12 +104,12 @@ class MockMgmtInstanceTest(trove_testtools.TestCase): compute_instance_id='compute_id_1', server_id='server_id_1', tenant_id='tenant_id_1', - server_status=rd_instance.ServiceStatuses. + server_status=srvstatus.ServiceStatuses. BUILDING.api_status, deleted=False) instance.save() service_status = InstanceServiceStatus( - rd_instance.ServiceStatuses.RUNNING, + srvstatus.ServiceStatuses.RUNNING, id=str(uuid.uuid4()), instance_id=instance.id, ) @@ -122,7 +128,7 @@ class TestNotificationTransformer(MockMgmtInstanceTest): @patch('trove.instance.models.LOG') def test_transformer(self, mock_logging): - status = rd_instance.ServiceStatuses.BUILDING.api_status + status = srvstatus.ServiceStatuses.BUILDING.api_status instance, service_status = self.build_db_instance( status, InstanceTasks.BUILDING) payloads = mgmtmodels.NotificationTransformer( @@ -184,7 +190,7 @@ class TestNovaNotificationTransformer(MockMgmtInstanceTest): Equals('unknown')) def test_transformer(self): - status = rd_instance.ServiceStatuses.BUILDING.api_status + status = srvstatus.ServiceStatuses.BUILDING.api_status instance, service_status = self.build_db_instance( status, InstanceTasks.BUILDING) @@ -223,7 +229,7 @@ class TestNovaNotificationTransformer(MockMgmtInstanceTest): @patch('trove.extensions.mgmt.instances.models.LOG') def test_transformer_invalid_datastore_manager(self, mock_logging): - status = rd_instance.ServiceStatuses.BUILDING.api_status + status = srvstatus.ServiceStatuses.BUILDING.api_status instance, service_status = self.build_db_instance( status, InstanceTasks.BUILDING) version = datastore_models.DBDatastoreVersion.get_by( @@ -268,9 +274,9 @@ class TestNovaNotificationTransformer(MockMgmtInstanceTest): self.addCleanup(self.do_cleanup, instance, service_status) def test_transformer_shutdown_instance(self): - status = rd_instance.ServiceStatuses.SHUTDOWN.api_status + status = srvstatus.ServiceStatuses.SHUTDOWN.api_status instance, service_status = self.build_db_instance(status) - service_status.set_status(rd_instance.ServiceStatuses.SHUTDOWN) + service_status.set_status(srvstatus.ServiceStatuses.SHUTDOWN) server = MagicMock(spec=Server) server.user_id = 'test_user_id' @@ -296,9 +302,9 @@ class TestNovaNotificationTransformer(MockMgmtInstanceTest): self.addCleanup(self.do_cleanup, instance, service_status) def test_transformer_no_nova_instance(self): - status = rd_instance.ServiceStatuses.SHUTDOWN.api_status + status = srvstatus.ServiceStatuses.SHUTDOWN.api_status instance, service_status = self.build_db_instance(status) - service_status.set_status(rd_instance.ServiceStatuses.SHUTDOWN) + service_status.set_status(srvstatus.ServiceStatuses.SHUTDOWN) mgmt_instance = mgmtmodels.SimpleMgmtInstance(self.context, instance, None, @@ -321,7 +327,7 @@ class TestNovaNotificationTransformer(MockMgmtInstanceTest): self.addCleanup(self.do_cleanup, instance, service_status) def test_transformer_flavor_cache(self): - status = rd_instance.ServiceStatuses.BUILDING.api_status + status = srvstatus.ServiceStatuses.BUILDING.api_status instance, service_status = self.build_db_instance( status, InstanceTasks.BUILDING) @@ -366,7 +372,7 @@ class TestMgmtInstanceTasks(MockMgmtInstanceTest): super(TestMgmtInstanceTasks, cls).setUpClass() def test_public_exists_events(self): - status = rd_instance.ServiceStatuses.BUILDING.api_status + status = srvstatus.ServiceStatuses.BUILDING.api_status instance, service_status = self.build_db_instance( status, task_status=InstanceTasks.BUILDING) server = MagicMock(spec=Server) @@ -443,7 +449,7 @@ class TestMgmtInstanceDeleted(MockMgmtInstanceTest): class TestMgmtInstancePing(MockMgmtInstanceTest): def test_rpc_ping(self): - status = rd_instance.ServiceStatuses.RUNNING.api_status + status = srvstatus.ServiceStatuses.RUNNING.api_status instance, service_status = self.build_db_instance( status, task_status=InstanceTasks.NONE) mgmt_instance = mgmtmodels.MgmtInstance(instance, diff --git a/trove/tests/unittests/taskmanager/test_clusters.py b/trove/tests/unittests/taskmanager/test_clusters.py index 1b6b7b52a9..9fa9e0cdd9 100644 --- a/trove/tests/unittests/taskmanager/test_clusters.py +++ b/trove/tests/unittests/taskmanager/test_clusters.py @@ -21,17 +21,16 @@ from mock import patch from trove.cluster.models import ClusterTasks as ClusterTaskStatus from trove.cluster.models import DBCluster +from trove.common import utils from trove.common.strategies.cluster.experimental.mongodb.taskmanager import ( MongoDbClusterTasks as ClusterTasks) -from trove.common import utils from trove.datastore import models as datastore_models from trove.instance.models import BaseInstance from trove.instance.models import DBInstance from trove.instance.models import Instance from trove.instance.models import InstanceServiceStatus from trove.instance.models import InstanceTasks -# from trove.taskmanager.models import BuiltInstanceTasks -from trove.taskmanager.models import ServiceStatuses +from trove.instance.service_status import ServiceStatuses from trove.tests.unittests import trove_testtools diff --git a/trove/tests/unittests/taskmanager/test_galera_clusters.py b/trove/tests/unittests/taskmanager/test_galera_clusters.py index 1b001453f9..5d32ce9e94 100644 --- a/trove/tests/unittests/taskmanager/test_galera_clusters.py +++ b/trove/tests/unittests/taskmanager/test_galera_clusters.py @@ -29,7 +29,7 @@ from trove.instance.models import DBInstance from trove.instance.models import Instance from trove.instance.models import InstanceServiceStatus from trove.instance.models import InstanceTasks -from trove.taskmanager.models import ServiceStatuses +from trove.instance.service_status import ServiceStatuses from trove.tests.unittests import trove_testtools from trove.tests.unittests.util import util diff --git a/trove/tests/unittests/taskmanager/test_models.py b/trove/tests/unittests/taskmanager/test_models.py index f3315f71b7..cbf6c74bf6 100644 --- a/trove/tests/unittests/taskmanager/test_models.py +++ b/trove/tests/unittests/taskmanager/test_models.py @@ -16,29 +16,34 @@ from tempfile import NamedTemporaryFile from unittest import mock from cinderclient import exceptions as cinder_exceptions -import cinderclient.v2.client as cinderclient from cinderclient.v2 import volumes as cinderclient_volumes -from mock import Mock, MagicMock, patch, PropertyMock, call +import cinderclient.v2.client as cinderclient +from mock import call +from mock import MagicMock +from mock import Mock +from mock import patch +from mock import PropertyMock import neutronclient.v2_0.client as neutronclient from novaclient import exceptions as nova_exceptions import novaclient.v2.flavors import novaclient.v2.servers from oslo_config import cfg from swiftclient.client import ClientException -from testtools.matchers import Equals, Is +from testtools.matchers import Equals +from testtools.matchers import Is -import trove.backup.models +from trove import rpc from trove.backup import models as backup_models from trove.backup import state +import trove.backup.models +from trove.common import timeutils +from trove.common import utils import trove.common.context from trove.common.exception import GuestError from trove.common.exception import PollTimeOut from trove.common.exception import TroveError -from trove.common.instance import ServiceStatuses from trove.common.notification import TroveInstanceModifyVolume import trove.common.template as template -from trove.common import timeutils -from trove.common import utils from trove.datastore import models as datastore_models import trove.db.models from trove.extensions.common import models as common_models @@ -48,8 +53,8 @@ from trove.instance.models import BaseInstance from trove.instance.models import DBInstance from trove.instance.models import InstanceServiceStatus from trove.instance.models import InstanceStatus +from trove.instance.service_status import ServiceStatuses from trove.instance.tasks import InstanceTasks -from trove import rpc from trove.taskmanager import models as taskmanager_models from trove.tests.unittests import trove_testtools from trove.tests.unittests.util import util @@ -962,27 +967,20 @@ class BuiltInstanceTasksTest(trove_testtools.TestCase): self.instance_task.demote_replication_master() self.instance_task._guest.demote_replication_master.assert_any_call() - @patch.multiple(taskmanager_models.BuiltInstanceTasks, - get_injected_files=Mock(return_value="the-files")) - def test_upgrade(self, *args): - pre_rebuild_server = self.instance_task.server - dsv = Mock(image_id='foo_image') - mock_volume = Mock(attachments=[{'device': '/dev/mock_dev'}]) - with patch.object(self.instance_task._volume_client.volumes, "get", - Mock(return_value=mock_volume)): - mock_server = Mock(status='ACTIVE') - with patch.object(self.instance_task._nova_client.servers, - 'get', Mock(return_value=mock_server)): - with patch.multiple(self.instance_task._guest, - pre_upgrade=Mock(return_value={}), - post_upgrade=Mock()): - self.instance_task.upgrade(dsv) + @patch('trove.taskmanager.models.BuiltInstanceTasks.set_service_status') + @patch('trove.taskmanager.models.BuiltInstanceTasks.is_service_healthy') + @patch('trove.taskmanager.models.BuiltInstanceTasks.reset_task_status') + def test_upgrade(self, mock_resetstatus, mock_check, mock_setstatus): + dsv = MagicMock() + attrs = {'name': 'new_version'} + dsv.configure_mock(**attrs) + mock_check.return_value = True + self.instance_task._guest.pre_upgrade.return_value = {} - self.instance_task._guest.pre_upgrade.assert_called_with() - pre_rebuild_server.rebuild.assert_called_with( - dsv.image_id, files="the-files") - self.instance_task._guest.post_upgrade.assert_called_with( - mock_volume.attachments[0]) + self.instance_task.upgrade(dsv) + + self.instance_task._guest.upgrade.assert_called_once_with( + {'datastore_version': 'new_version'}) def test_fix_device_path(self): self.assertEqual("/dev/vdb", self.instance_task. diff --git a/trove/tests/unittests/taskmanager/test_vertica_clusters.py b/trove/tests/unittests/taskmanager/test_vertica_clusters.py index 8d0a14e1eb..d066f36e02 100644 --- a/trove/tests/unittests/taskmanager/test_vertica_clusters.py +++ b/trove/tests/unittests/taskmanager/test_vertica_clusters.py @@ -16,6 +16,7 @@ import datetime from mock import Mock from mock import patch +from trove import rpc from trove.cluster.models import ClusterTasks as ClusterTaskStatus from trove.cluster.models import DBCluster import trove.common.context as context @@ -32,8 +33,7 @@ from trove.instance.models import DBInstance from trove.instance.models import Instance from trove.instance.models import InstanceServiceStatus from trove.instance.models import InstanceTasks -from trove import rpc -from trove.taskmanager.models import ServiceStatuses +from trove.instance.service_status import ServiceStatuses from trove.tests.unittests import trove_testtools