Fixes resizes for volumes attached to active Nova servers
Currently, Trove fails to resize volumes attached to active Nova servers. It also skips crucial steps such as unmounting and mounting the volume and resizing the filesystem. When extending a volume attached to an active Nova server, the guest must be stopped, the volume unmounted, detached, and extended. Once the underlying volume has been extended the volume should be reattached, have the file system resized, mounted, and then the guest should be restarted. If all operations complete successfully the task status should be set to NONE and a usage event sent, reflecting the size of the volume as reported by Cinder. This fixes extending volumes attached to active Nova servers to ensure a proper resize is performed and the task status is set to NONE and a usage event is sent after all resize operations complete successfully. Unit tests are also added. Change-Id: I9a5785f539a2ee138ce1ded3cbd795ef91d66400 Closes-Bug: #1259976
This commit is contained in:
parent
b4100f7f30
commit
522f6df6ad
@ -314,3 +314,18 @@ def correct_id_with_req(id, request):
|
|||||||
|
|
||||||
def generate_random_password(password_length=CONF.default_password_length):
|
def generate_random_password(password_length=CONF.default_password_length):
|
||||||
return passlib_utils.generate_password(size=password_length)
|
return passlib_utils.generate_password(size=password_length)
|
||||||
|
|
||||||
|
|
||||||
|
def try_recover(func):
|
||||||
|
def _decorator(*args, **kwargs):
|
||||||
|
recover_func = kwargs.pop("recover_func", None)
|
||||||
|
try:
|
||||||
|
func(*args, **kwargs)
|
||||||
|
except Exception:
|
||||||
|
if recover_func is not None:
|
||||||
|
recover_func(func)
|
||||||
|
else:
|
||||||
|
LOG.debug(_("No recovery method defined for %(func)s") % {
|
||||||
|
'func': func.__name__})
|
||||||
|
raise
|
||||||
|
return _decorator
|
||||||
|
@ -280,3 +280,24 @@ class API(proxy.RpcProxy):
|
|||||||
"for Instance %(instance_id)s") %
|
"for Instance %(instance_id)s") %
|
||||||
{'backup_id': backup_info['id'], 'instance_id': self.id})
|
{'backup_id': backup_info['id'], 'instance_id': self.id})
|
||||||
self._cast("create_backup", backup_info=backup_info)
|
self._cast("create_backup", backup_info=backup_info)
|
||||||
|
|
||||||
|
def mount_volume(self, device_path=None, mount_point=None):
|
||||||
|
"""Mount the volume"""
|
||||||
|
LOG.debug(_("Mount volume %(mount)s on instance %(id)s") % {
|
||||||
|
'mount': mount_point, 'id': self.id})
|
||||||
|
self._call("mount_volume", AGENT_LOW_TIMEOUT,
|
||||||
|
device_path=device_path, mount_point=mount_point)
|
||||||
|
|
||||||
|
def unmount_volume(self, device_path=None, mount_point=None):
|
||||||
|
"""Unmount the volume"""
|
||||||
|
LOG.debug(_("Unmount volume %(device)s on instance %(id)s") % {
|
||||||
|
'device': device_path, 'id': self.id})
|
||||||
|
self._call("unmount_volume", AGENT_LOW_TIMEOUT,
|
||||||
|
device_path=device_path, mount_point=mount_point)
|
||||||
|
|
||||||
|
def resize_fs(self, device_path=None, mount_point=None):
|
||||||
|
"""Resize the filesystem"""
|
||||||
|
LOG.debug(_("Resize device %(device)s on instance %(id)s") % {
|
||||||
|
'device': device_path, 'id': self.id})
|
||||||
|
self._call("resize_fs", AGENT_LOW_TIMEOUT, device_path=device_path,
|
||||||
|
mount_point=mount_point)
|
||||||
|
@ -175,3 +175,18 @@ class Manager(periodic_task.PeriodicTasks):
|
|||||||
backup task, location, type, and other data.
|
backup task, location, type, and other data.
|
||||||
"""
|
"""
|
||||||
backup.backup(context, backup_info)
|
backup.backup(context, backup_info)
|
||||||
|
|
||||||
|
def mount_volume(self, context, device_path=None, mount_point=None):
|
||||||
|
device = volume.VolumeDevice(device_path)
|
||||||
|
device.mount(mount_point, write_to_fstab=False)
|
||||||
|
LOG.debug(_("Mounted the volume."))
|
||||||
|
|
||||||
|
def unmount_volume(self, context, device_path=None, mount_point=None):
|
||||||
|
device = volume.VolumeDevice(device_path)
|
||||||
|
device.unmount(mount_point)
|
||||||
|
LOG.debug(_("Unmounted the volume."))
|
||||||
|
|
||||||
|
def resize_fs(self, context, device_path=None, mount_point=None):
|
||||||
|
device = volume.VolumeDevice(device_path)
|
||||||
|
device.resize_fs(mount_point)
|
||||||
|
LOG.debug(_("Resized the filesystem"))
|
||||||
|
@ -37,13 +37,13 @@ class VolumeDevice(object):
|
|||||||
|
|
||||||
def migrate_data(self, mysql_base):
|
def migrate_data(self, mysql_base):
|
||||||
"""Synchronize the data from the mysql directory to the new volume """
|
"""Synchronize the data from the mysql directory to the new volume """
|
||||||
self._tmp_mount(TMP_MOUNT_POINT)
|
self.mount(TMP_MOUNT_POINT, write_to_fstab=False)
|
||||||
if not mysql_base[-1] == '/':
|
if not mysql_base[-1] == '/':
|
||||||
mysql_base = "%s/" % mysql_base
|
mysql_base = "%s/" % mysql_base
|
||||||
utils.execute("sudo", "rsync", "--safe-links", "--perms",
|
utils.execute("sudo", "rsync", "--safe-links", "--perms",
|
||||||
"--recursive", "--owner", "--group", "--xattrs",
|
"--recursive", "--owner", "--group", "--xattrs",
|
||||||
"--sparse", mysql_base, TMP_MOUNT_POINT)
|
"--sparse", mysql_base, TMP_MOUNT_POINT)
|
||||||
self.unmount()
|
self.unmount(TMP_MOUNT_POINT)
|
||||||
|
|
||||||
def _check_device_exists(self):
|
def _check_device_exists(self):
|
||||||
"""Check that the device path exists.
|
"""Check that the device path exists.
|
||||||
@ -91,31 +91,29 @@ class VolumeDevice(object):
|
|||||||
self._format()
|
self._format()
|
||||||
self._check_format()
|
self._check_format()
|
||||||
|
|
||||||
def mount(self, mount_point):
|
def mount(self, mount_point, write_to_fstab=True):
|
||||||
"""Mounts, and writes to fstab."""
|
"""Mounts, and writes to fstab."""
|
||||||
mount_point = VolumeMountPoint(self.device_path, mount_point)
|
mount_point = VolumeMountPoint(self.device_path, mount_point)
|
||||||
mount_point.mount()
|
mount_point.mount()
|
||||||
|
if write_to_fstab:
|
||||||
mount_point.write_to_fstab()
|
mount_point.write_to_fstab()
|
||||||
|
|
||||||
#TODO(tim.simpson): Are we using this?
|
def resize_fs(self, mount_point):
|
||||||
def resize_fs(self):
|
|
||||||
"""Resize the filesystem on the specified device"""
|
"""Resize the filesystem on the specified device"""
|
||||||
self._check_device_exists()
|
self._check_device_exists()
|
||||||
try:
|
try:
|
||||||
|
# check if the device is mounted at mount_point before e2fsck
|
||||||
|
if not os.path.ismount(mount_point):
|
||||||
|
utils.execute("sudo", "e2fsck", "-f", "-n", self.device_path)
|
||||||
utils.execute("sudo", "resize2fs", self.device_path)
|
utils.execute("sudo", "resize2fs", self.device_path)
|
||||||
except ProcessExecutionError as err:
|
except ProcessExecutionError as err:
|
||||||
LOG.error(err)
|
LOG.error(err)
|
||||||
raise GuestError("Error resizing the filesystem: %s" %
|
raise GuestError("Error resizing the filesystem: %s" %
|
||||||
self.device_path)
|
self.device_path)
|
||||||
|
|
||||||
def _tmp_mount(self, mount_point):
|
def unmount(self, mount_point):
|
||||||
"""Mounts, but doesn't save to fstab."""
|
if os.path.exists(mount_point):
|
||||||
mount_point = VolumeMountPoint(self.device_path, mount_point)
|
cmd = "sudo umount %s" % mount_point
|
||||||
mount_point.mount() # Don't save to fstab.
|
|
||||||
|
|
||||||
def unmount(self):
|
|
||||||
if os.path.exists(self.device_path):
|
|
||||||
cmd = "sudo umount %s" % self.device_path
|
|
||||||
child = pexpect.spawn(cmd)
|
child = pexpect.spawn(cmd)
|
||||||
child.expect(pexpect.EOF)
|
child.expect(pexpect.EOF)
|
||||||
|
|
||||||
@ -131,7 +129,7 @@ class VolumeMountPoint(object):
|
|||||||
def mount(self):
|
def mount(self):
|
||||||
if not os.path.exists(self.mount_point):
|
if not os.path.exists(self.mount_point):
|
||||||
utils.execute("sudo", "mkdir", "-p", self.mount_point)
|
utils.execute("sudo", "mkdir", "-p", self.mount_point)
|
||||||
LOG.debug("Adding volume. Device path:%s, mount_point:%s, "
|
LOG.debug("Mounting volume. Device path:%s, mount_point:%s, "
|
||||||
"volume_type:%s, mount options:%s" %
|
"volume_type:%s, mount options:%s" %
|
||||||
(self.device_path, self.mount_point, self.volume_fstype,
|
(self.device_path, self.mount_point, self.volume_fstype,
|
||||||
self.mount_options))
|
self.mount_options))
|
||||||
|
@ -23,6 +23,7 @@ from trove.backup import models as bkup_models
|
|||||||
from trove.common import cfg
|
from trove.common import cfg
|
||||||
from trove.common import template
|
from trove.common import template
|
||||||
from trove.common import utils
|
from trove.common import utils
|
||||||
|
from trove.common.utils import try_recover
|
||||||
from trove.common.exception import GuestError
|
from trove.common.exception import GuestError
|
||||||
from trove.common.exception import GuestTimeout
|
from trove.common.exception import GuestTimeout
|
||||||
from trove.common.exception import PollTimeOut
|
from trove.common.exception import PollTimeOut
|
||||||
@ -695,95 +696,11 @@ class BuiltInstanceTasks(BuiltInstance, NotifyMixin, ConfigurationMixin):
|
|||||||
server=old_server)
|
server=old_server)
|
||||||
LOG.debug(_("end _delete_resources for id: %s") % self.id)
|
LOG.debug(_("end _delete_resources for id: %s") % self.id)
|
||||||
|
|
||||||
def _resize_active_volume(self, new_size):
|
|
||||||
try:
|
|
||||||
LOG.debug(_("Instance %s calling stop_db...") % self.server.id)
|
|
||||||
self.guest.stop_db()
|
|
||||||
|
|
||||||
LOG.debug(_("Detach volume %(vol_id)s from instance %(id)s") %
|
|
||||||
{'vol_id': self.volume_id, 'id': self.server.id})
|
|
||||||
self.volume_client.volumes.detach(self.volume_id)
|
|
||||||
|
|
||||||
utils.poll_until(
|
|
||||||
lambda: self.volume_client.volumes.get(self.volume_id),
|
|
||||||
lambda volume: volume.status == 'available',
|
|
||||||
sleep_time=2,
|
|
||||||
time_out=CONF.volume_time_out)
|
|
||||||
|
|
||||||
LOG.debug(_("Successfully detach volume %s") % self.volume_id)
|
|
||||||
except Exception as e:
|
|
||||||
LOG.debug(_("end _resize_active_volume for id: %s") %
|
|
||||||
self.server.id)
|
|
||||||
LOG.exception(_("Failed to detach volume %(volume_id)s "
|
|
||||||
"instance %(id)s: %(e)s") %
|
|
||||||
{'volume_id': self.volume_id, 'id':
|
|
||||||
self.server.id, 'e': str(e)})
|
|
||||||
self.restart()
|
|
||||||
raise
|
|
||||||
|
|
||||||
self._do_resize(new_size)
|
|
||||||
self.volume_client.volumes.attach(self.server.id, self.volume_id)
|
|
||||||
LOG.debug(_("end _resize_active_volume for id: %s") % self.server.id)
|
|
||||||
self.restart()
|
|
||||||
|
|
||||||
def _do_resize(self, new_size):
|
|
||||||
try:
|
|
||||||
self.volume_client.volumes.extend(self.volume_id, new_size)
|
|
||||||
except cinder_exceptions.ClientException:
|
|
||||||
LOG.exception(_("Error encountered trying to rescan or resize the "
|
|
||||||
"attached volume filesystem for volume: "
|
|
||||||
"%s") % self.volume_id)
|
|
||||||
raise
|
|
||||||
|
|
||||||
try:
|
|
||||||
volume = self.volume_client.volumes.get(self.volume_id)
|
|
||||||
if not volume:
|
|
||||||
raise (cinder_exceptions.
|
|
||||||
ClientException(_('Failed to get volume with '
|
|
||||||
'id: %(id)s') %
|
|
||||||
{'id': self.volume_id}))
|
|
||||||
utils.poll_until(
|
|
||||||
lambda: self.volume_client.volumes.get(self.volume_id),
|
|
||||||
lambda volume: volume.size == int(new_size),
|
|
||||||
sleep_time=2,
|
|
||||||
time_out=CONF.volume_time_out)
|
|
||||||
self.update_db(volume_size=new_size)
|
|
||||||
except PollTimeOut:
|
|
||||||
LOG.error(_("Timeout trying to rescan or resize the attached "
|
|
||||||
"volume filesystem for volume %(vol_id)s of "
|
|
||||||
"instance: %(id)s") %
|
|
||||||
{'vol_id': self.volume_id, 'id': self.id})
|
|
||||||
except Exception as e:
|
|
||||||
LOG.exception(_("Error encountered trying to rescan or resize the "
|
|
||||||
"attached volume filesystem of volume %(vol_id)s of "
|
|
||||||
"instance %(id)s: %(e)s") %
|
|
||||||
{'vol_id': self.volume_id, 'id': self.id, 'e': e})
|
|
||||||
finally:
|
|
||||||
self.update_db(task_status=inst_models.InstanceTasks.NONE)
|
|
||||||
|
|
||||||
def resize_volume(self, new_size):
|
def resize_volume(self, new_size):
|
||||||
LOG.debug(_("begin resize_volume for id: %s") % self.id)
|
LOG.debug(_("begin resize_volume for instance: %s") % self.id)
|
||||||
old_volume_size = self.volume_size
|
action = ResizeVolumeAction(self, self.volume_size, new_size)
|
||||||
new_size = int(new_size)
|
action.execute()
|
||||||
LOG.debug(_("%(gt)s: Resizing instance %(instance_id)s volume for "
|
LOG.debug(_("end resize_volume for instance: %s") % self.id)
|
||||||
"server %(server_id)s from %(old_volume_size)s to "
|
|
||||||
"%(new_size)r GB")
|
|
||||||
% {'gt': greenthread.getcurrent(),
|
|
||||||
'instance_id': self.id,
|
|
||||||
'server_id': self.server.id,
|
|
||||||
'old_volume_size': old_volume_size,
|
|
||||||
'new_size': new_size})
|
|
||||||
|
|
||||||
if self.server.status == 'active':
|
|
||||||
self._resize_active_volume(new_size)
|
|
||||||
else:
|
|
||||||
self._do_resize(new_size)
|
|
||||||
|
|
||||||
self.send_usage_event('modify_volume', old_volume_size=old_volume_size,
|
|
||||||
launched_at=timeutils.isotime(self.updated),
|
|
||||||
modify_at=timeutils.isotime(self.updated),
|
|
||||||
volume_size=new_size)
|
|
||||||
LOG.debug(_("end resize_volume for id: %s") % self.id)
|
|
||||||
|
|
||||||
def resize_flavor(self, old_flavor, new_flavor):
|
def resize_flavor(self, old_flavor, new_flavor):
|
||||||
action = ResizeAction(self, old_flavor, new_flavor)
|
action = ResizeAction(self, old_flavor, new_flavor)
|
||||||
@ -918,6 +835,222 @@ class BackupTasks(object):
|
|||||||
backup.delete()
|
backup.delete()
|
||||||
|
|
||||||
|
|
||||||
|
class ResizeVolumeAction(ConfigurationMixin):
|
||||||
|
"""Performs volume resize action."""
|
||||||
|
|
||||||
|
def __init__(self, instance, old_size, new_size):
|
||||||
|
self.instance = instance
|
||||||
|
self.old_size = int(old_size)
|
||||||
|
self.new_size = int(new_size)
|
||||||
|
|
||||||
|
def _fail(self, orig_func):
|
||||||
|
LOG.exception(_("%(func)s encountered an error when attempting to "
|
||||||
|
"resize the volume for instance %(id)s. Setting service "
|
||||||
|
"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.save()
|
||||||
|
|
||||||
|
def _recover_restart(self, orig_func):
|
||||||
|
LOG.exception(_("%(func)s encountered an error when attempting to "
|
||||||
|
"resize the volume for instance %(id)s. Trying to "
|
||||||
|
"recover by restarting the guest.") % {
|
||||||
|
'func': orig_func.__name__,
|
||||||
|
'id': self.instance.id})
|
||||||
|
self.instance.restart()
|
||||||
|
|
||||||
|
def _recover_mount_restart(self, orig_func):
|
||||||
|
LOG.exception(_("%(func)s encountered an error when attempting to "
|
||||||
|
"resize the volume for instance %(id)s. Trying to "
|
||||||
|
"recover by mounting the volume and then restarting the "
|
||||||
|
"guest.") % {'func': orig_func.__name__,
|
||||||
|
'id': self.instance.id})
|
||||||
|
self._mount_volume()
|
||||||
|
self.instance.restart()
|
||||||
|
|
||||||
|
def _recover_full(self, orig_func):
|
||||||
|
LOG.exception(_("%(func)s encountered an error when attempting to "
|
||||||
|
"resize the volume for instance %(id)s. Trying to "
|
||||||
|
"recover by attaching and mounting the volume and then "
|
||||||
|
"restarting the guest.") % {'func': orig_func.__name__,
|
||||||
|
'id': self.instance.id})
|
||||||
|
self._attach_volume()
|
||||||
|
self._mount_volume()
|
||||||
|
self.instance.restart()
|
||||||
|
|
||||||
|
def _stop_db(self):
|
||||||
|
LOG.debug(_("Instance %s calling stop_db.") % self.instance.id)
|
||||||
|
self.instance.guest.stop_db()
|
||||||
|
|
||||||
|
@try_recover
|
||||||
|
def _unmount_volume(self):
|
||||||
|
LOG.debug(_("Unmounting the volume on instance %(id)s") % {
|
||||||
|
'id': self.instance.id})
|
||||||
|
self.instance.guest.unmount_volume(device_path=CONF.device_path,
|
||||||
|
mount_point=CONF.mount_point)
|
||||||
|
LOG.debug(_("Successfully unmounted the volume %(vol_id)s for "
|
||||||
|
"instance %(id)s") % {'vol_id': self.instance.volume_id,
|
||||||
|
'id': self.instance.id})
|
||||||
|
|
||||||
|
@try_recover
|
||||||
|
def _detach_volume(self):
|
||||||
|
LOG.debug(_("Detach volume %(vol_id)s from instance %(id)s") % {
|
||||||
|
'vol_id': self.instance.volume_id,
|
||||||
|
'id': self.instance.id})
|
||||||
|
self.instance.volume_client.volumes.detach(self.instance.volume_id)
|
||||||
|
|
||||||
|
def volume_available():
|
||||||
|
volume = self.instance.volume_client.volumes.get(
|
||||||
|
self.instance.volume_id)
|
||||||
|
return volume.status == 'available'
|
||||||
|
utils.poll_until(volume_available,
|
||||||
|
sleep_time=2,
|
||||||
|
time_out=CONF.volume_time_out)
|
||||||
|
|
||||||
|
LOG.debug(_("Successfully detached volume %(vol_id)s from instance "
|
||||||
|
"%(id)s") % {'vol_id': self.instance.volume_id,
|
||||||
|
'id': self.instance.id})
|
||||||
|
|
||||||
|
@try_recover
|
||||||
|
def _attach_volume(self):
|
||||||
|
LOG.debug(_("Attach volume %(vol_id)s to instance %(id)s at "
|
||||||
|
"%(dev)s") % {'vol_id': self.instance.volume_id,
|
||||||
|
'id': self.instance.id, 'dev': CONF.device_path})
|
||||||
|
self.instance.volume_client.volumes.attach(self.instance.volume_id,
|
||||||
|
self.instance.server.id,
|
||||||
|
CONF.device_path)
|
||||||
|
|
||||||
|
def volume_in_use():
|
||||||
|
volume = self.instance.volume_client.volumes.get(
|
||||||
|
self.instance.volume_id)
|
||||||
|
return volume.status == 'in-use'
|
||||||
|
utils.poll_until(volume_in_use,
|
||||||
|
sleep_time=2,
|
||||||
|
time_out=CONF.volume_time_out)
|
||||||
|
|
||||||
|
LOG.debug(_("Successfully attached volume %(vol_id)s to instance "
|
||||||
|
"%(id)s") % {'vol_id': self.instance.volume_id,
|
||||||
|
'id': self.instance.id})
|
||||||
|
|
||||||
|
@try_recover
|
||||||
|
def _resize_fs(self):
|
||||||
|
LOG.debug(_("Resizing the filesystem for instance %(id)s") % {
|
||||||
|
'id': self.instance.id})
|
||||||
|
self.instance.guest.resize_fs(device_path=CONF.device_path,
|
||||||
|
mount_point=CONF.mount_point)
|
||||||
|
LOG.debug(_("Successfully resized volume %(vol_id)s filesystem for "
|
||||||
|
"instance %(id)s") % {'vol_id': self.instance.volume_id,
|
||||||
|
'id': self.instance.id})
|
||||||
|
|
||||||
|
@try_recover
|
||||||
|
def _mount_volume(self):
|
||||||
|
LOG.debug(_("Mount the volume on instance %(id)s") % {
|
||||||
|
'id': self.instance.id})
|
||||||
|
self.instance.guest.mount_volume(device_path=CONF.device_path,
|
||||||
|
mount_point=CONF.mount_point)
|
||||||
|
LOG.debug(_("Successfully mounted the volume %(vol_id)s on instance "
|
||||||
|
"%(id)s") % {'vol_id': self.instance.volume_id,
|
||||||
|
'id': self.instance.id})
|
||||||
|
|
||||||
|
@try_recover
|
||||||
|
def _extend(self):
|
||||||
|
LOG.debug(_("Extending volume %(vol_id)s for instance %(id)s to "
|
||||||
|
"size %(size)s") % {'vol_id': self.instance.volume_id,
|
||||||
|
'id': self.instance.id, 'size': self.new_size})
|
||||||
|
self.instance.volume_client.volumes.extend(self.instance.volume_id,
|
||||||
|
self.new_size)
|
||||||
|
LOG.debug(_("Successfully extended the volume %(vol_id)s for instance "
|
||||||
|
"%(id)s") % {'vol_id': self.instance.volume_id,
|
||||||
|
'id': self.instance.id})
|
||||||
|
|
||||||
|
def _verify_extend(self):
|
||||||
|
try:
|
||||||
|
volume = self.instance.volume_client.volumes.get(
|
||||||
|
self.instance.volume_id)
|
||||||
|
if not volume:
|
||||||
|
msg = (_('Failed to get volume %(vol_id)s') % {
|
||||||
|
'vol_id': self.instance.volume_id})
|
||||||
|
raise cinder_exceptions.ClientException(msg)
|
||||||
|
|
||||||
|
def volume_is_new_size():
|
||||||
|
volume = self.instance.volume_client.volumes.get(
|
||||||
|
self.instance.volume_id)
|
||||||
|
return volume.size == self.new_size
|
||||||
|
utils.poll_until(volume_is_new_size,
|
||||||
|
sleep_time=2,
|
||||||
|
time_out=CONF.volume_time_out)
|
||||||
|
|
||||||
|
self.instance.update_db(volume_size=self.new_size)
|
||||||
|
except PollTimeOut:
|
||||||
|
LOG.exception(_("Timeout trying to extend the volume %(vol_id)s "
|
||||||
|
"for instance %(id)s") % {
|
||||||
|
'vol_id': self.instance.volume_id,
|
||||||
|
'id': self.instance.id})
|
||||||
|
volume = self.instance.volume_client.volumes.get(
|
||||||
|
self.instance.volume_id)
|
||||||
|
if volume.status == 'extending':
|
||||||
|
self._fail(self._verify_extend)
|
||||||
|
elif volume.size != self.new_size:
|
||||||
|
self.instance.update_db(volume_size=volume.size)
|
||||||
|
self._recover_full(self._verify_extend)
|
||||||
|
raise
|
||||||
|
except Exception:
|
||||||
|
LOG.exception(_("Error encountered trying to verify extend for "
|
||||||
|
"the volume %(vol_id)s for instance %(id)s") % {
|
||||||
|
'vol_id': self.instance.volume_id,
|
||||||
|
'id': self.instance.id})
|
||||||
|
self._recover_full(self._verify_extend)
|
||||||
|
raise
|
||||||
|
|
||||||
|
def _resize_active_volume(self):
|
||||||
|
LOG.debug(_("begin _resize_active_volume for id: %(id)s") % {
|
||||||
|
'id': self.instance.id})
|
||||||
|
self._stop_db()
|
||||||
|
self._unmount_volume(recover_func=self._recover_restart)
|
||||||
|
self._detach_volume(recover_func=self._recover_mount_restart)
|
||||||
|
self._extend(recover_func=self._recover_full)
|
||||||
|
self._verify_extend()
|
||||||
|
# if anything fails after this point, recovery is futile
|
||||||
|
self._attach_volume(recover_func=self._fail)
|
||||||
|
self._resize_fs(recover_func=self._fail)
|
||||||
|
self._mount_volume(recover_func=self._fail)
|
||||||
|
self.instance.restart()
|
||||||
|
LOG.debug(_("end _resize_active_volume for id: %(id)s") % {
|
||||||
|
'id': self.instance.id})
|
||||||
|
|
||||||
|
def execute(self):
|
||||||
|
LOG.debug(_("%(gt)s: Resizing instance %(id)s volume for server "
|
||||||
|
"%(server_id)s from %(old_volume_size)s to "
|
||||||
|
"%(new_size)r GB") % {'gt': greenthread.getcurrent(),
|
||||||
|
'id': self.instance.id,
|
||||||
|
'server_id': self.instance.server.id,
|
||||||
|
'old_volume_size': self.old_size,
|
||||||
|
'new_size': self.new_size})
|
||||||
|
|
||||||
|
if self.instance.server.status == InstanceStatus.ACTIVE:
|
||||||
|
self._resize_active_volume()
|
||||||
|
self.instance.update_db(task_status=inst_models.InstanceTasks.NONE)
|
||||||
|
# send usage event for size reported by cinder
|
||||||
|
volume = self.instance.volume_client.volumes.get(
|
||||||
|
self.instance.volume_id)
|
||||||
|
launched_time = timeutils.isotime(self.instance.updated)
|
||||||
|
modified_time = timeutils.isotime(self.instance.updated)
|
||||||
|
self.instance.send_usage_event('modify_volume',
|
||||||
|
old_volume_size=self.old_size,
|
||||||
|
launched_at=launched_time,
|
||||||
|
modify_at=modified_time,
|
||||||
|
volume_size=volume.size)
|
||||||
|
else:
|
||||||
|
self.instance.update_db(task_status=inst_models.InstanceTasks.NONE)
|
||||||
|
msg = _("Volume resize failed for instance %(id)s. The instance "
|
||||||
|
"must be in state %(state)s not %(inst_state)s.") % {
|
||||||
|
'id': self.instance.id,
|
||||||
|
'state': InstanceStatus.ACTIVE,
|
||||||
|
'inst_state': self.instance.server.status}
|
||||||
|
raise TroveError(msg)
|
||||||
|
|
||||||
|
|
||||||
class ResizeActionBase(ConfigurationMixin):
|
class ResizeActionBase(ConfigurationMixin):
|
||||||
"""Base class for executing a resize action."""
|
"""Base class for executing a resize action."""
|
||||||
|
|
||||||
|
@ -308,6 +308,15 @@ class FakeGuest(object):
|
|||||||
backup.save()
|
backup.save()
|
||||||
eventlet.spawn_after(1.0, finish_create_backup)
|
eventlet.spawn_after(1.0, finish_create_backup)
|
||||||
|
|
||||||
|
def mount_volume(self, device_path=None, mount_point=None):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def unmount_volume(self, device_path=None, mount_point=None):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def resize_fs(self, device_path=None, mount_point=None):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
def get_or_create(id):
|
def get_or_create(id):
|
||||||
if id not in DB:
|
if id not in DB:
|
||||||
|
@ -113,6 +113,7 @@ class FakeServer(object):
|
|||||||
for volume in self.volumes:
|
for volume in self.volumes:
|
||||||
info_vols.append({'id': volume.id})
|
info_vols.append({'id': volume.id})
|
||||||
volume.set_attachment(id)
|
volume.set_attachment(id)
|
||||||
|
volume.schedule_status("in-use", 1)
|
||||||
self.host = FAKE_HOSTS[0]
|
self.host = FAKE_HOSTS[0]
|
||||||
self.old_host = None
|
self.old_host = None
|
||||||
setattr(self, 'OS-EXT-AZ:availability_zone', 'nova')
|
setattr(self, 'OS-EXT-AZ:availability_zone', 'nova')
|
||||||
@ -512,6 +513,16 @@ class FakeVolumes(object):
|
|||||||
volume._current_status = "available"
|
volume._current_status = "available"
|
||||||
eventlet.spawn_after(1.0, finish_detach)
|
eventlet.spawn_after(1.0, finish_detach)
|
||||||
|
|
||||||
|
def attach(self, volume_id, server_id, device_path):
|
||||||
|
volume = self.get(volume_id)
|
||||||
|
|
||||||
|
if volume._current_status != "available":
|
||||||
|
raise Exception("Invalid volume status")
|
||||||
|
|
||||||
|
def finish_attach():
|
||||||
|
volume._current_status = "in-use"
|
||||||
|
eventlet.spawn_after(1.0, finish_attach)
|
||||||
|
|
||||||
|
|
||||||
class FakeAccount(object):
|
class FakeAccount(object):
|
||||||
|
|
||||||
|
@ -38,18 +38,19 @@ class VolumeDeviceTest(testtools.TestCase):
|
|||||||
def test_migrate_data(self):
|
def test_migrate_data(self):
|
||||||
origin_execute = utils.execute
|
origin_execute = utils.execute
|
||||||
utils.execute = Mock()
|
utils.execute = Mock()
|
||||||
|
origin_os_path_exists = os.path.exists
|
||||||
|
os.path.exists = Mock()
|
||||||
|
fake_spawn = _setUp_fake_spawn()
|
||||||
|
|
||||||
origin_tmp_mount = self.volumeDevice._tmp_mount
|
|
||||||
origin_unmount = self.volumeDevice.unmount
|
origin_unmount = self.volumeDevice.unmount
|
||||||
self.volumeDevice._tmp_mount = MagicMock()
|
|
||||||
self.volumeDevice.unmount = MagicMock()
|
self.volumeDevice.unmount = MagicMock()
|
||||||
self.volumeDevice.migrate_data('/')
|
self.volumeDevice.migrate_data('/')
|
||||||
|
self.assertEqual(1, fake_spawn.expect.call_count)
|
||||||
self.assertEqual(1, utils.execute.call_count)
|
self.assertEqual(1, utils.execute.call_count)
|
||||||
self.assertEqual(1, self.volumeDevice._tmp_mount.call_count)
|
|
||||||
self.assertEqual(1, self.volumeDevice.unmount.call_count)
|
self.assertEqual(1, self.volumeDevice.unmount.call_count)
|
||||||
utils.execute = origin_execute
|
utils.execute = origin_execute
|
||||||
self.volumeDevice._tmp_mount = origin_tmp_mount
|
|
||||||
self.volumeDevice.unmount = origin_unmount
|
self.volumeDevice.unmount = origin_unmount
|
||||||
|
os.path.exists = origin_os_path_exists
|
||||||
|
|
||||||
def test__check_device_exists(self):
|
def test__check_device_exists(self):
|
||||||
origin_execute = utils.execute
|
origin_execute = utils.execute
|
||||||
@ -98,6 +99,8 @@ class VolumeDeviceTest(testtools.TestCase):
|
|||||||
def test_mount(self):
|
def test_mount(self):
|
||||||
origin_ = volume.VolumeMountPoint.mount
|
origin_ = volume.VolumeMountPoint.mount
|
||||||
volume.VolumeMountPoint.mount = Mock()
|
volume.VolumeMountPoint.mount = Mock()
|
||||||
|
origin_os_path_exists = os.path.exists
|
||||||
|
os.path.exists = Mock()
|
||||||
origin_write_to_fstab = volume.VolumeMountPoint.write_to_fstab
|
origin_write_to_fstab = volume.VolumeMountPoint.write_to_fstab
|
||||||
volume.VolumeMountPoint.write_to_fstab = Mock()
|
volume.VolumeMountPoint.write_to_fstab = Mock()
|
||||||
|
|
||||||
@ -106,28 +109,24 @@ class VolumeDeviceTest(testtools.TestCase):
|
|||||||
self.assertEqual(1, volume.VolumeMountPoint.write_to_fstab.call_count)
|
self.assertEqual(1, volume.VolumeMountPoint.write_to_fstab.call_count)
|
||||||
volume.VolumeMountPoint.mount = origin_
|
volume.VolumeMountPoint.mount = origin_
|
||||||
volume.VolumeMountPoint.write_to_fstab = origin_write_to_fstab
|
volume.VolumeMountPoint.write_to_fstab = origin_write_to_fstab
|
||||||
|
os.path.exists = origin_os_path_exists
|
||||||
|
|
||||||
def test_resize_fs(self):
|
def test_resize_fs(self):
|
||||||
origin_check_device_exists = self.volumeDevice._check_device_exists
|
origin_check_device_exists = self.volumeDevice._check_device_exists
|
||||||
origin_execute = utils.execute
|
origin_execute = utils.execute
|
||||||
utils.execute = Mock()
|
utils.execute = Mock()
|
||||||
self.volumeDevice._check_device_exists = MagicMock()
|
self.volumeDevice._check_device_exists = MagicMock()
|
||||||
|
origin_os_path_exists = os.path.exists
|
||||||
|
os.path.exists = Mock()
|
||||||
|
|
||||||
self.volumeDevice.resize_fs()
|
self.volumeDevice.resize_fs('/mnt/volume')
|
||||||
|
|
||||||
self.assertEqual(1, self.volumeDevice._check_device_exists.call_count)
|
self.assertEqual(1, self.volumeDevice._check_device_exists.call_count)
|
||||||
self.assertEqual(1, utils.execute.call_count)
|
self.assertEqual(2, utils.execute.call_count)
|
||||||
self.volumeDevice._check_device_exists = origin_check_device_exists
|
self.volumeDevice._check_device_exists = origin_check_device_exists
|
||||||
|
os.path.exists = origin_os_path_exists
|
||||||
utils.execute = origin_execute
|
utils.execute = origin_execute
|
||||||
|
|
||||||
def test__tmp_mount(self):
|
|
||||||
origin_ = volume.VolumeMountPoint.mount
|
|
||||||
volume.VolumeMountPoint.mount = Mock()
|
|
||||||
|
|
||||||
self.volumeDevice._tmp_mount(Mock)
|
|
||||||
self.assertEqual(1, volume.VolumeMountPoint.mount.call_count)
|
|
||||||
volume.VolumeMountPoint.mount = origin_
|
|
||||||
|
|
||||||
def test_unmount_positive(self):
|
def test_unmount_positive(self):
|
||||||
self._test_unmount()
|
self._test_unmount()
|
||||||
|
|
||||||
@ -139,7 +138,7 @@ class VolumeDeviceTest(testtools.TestCase):
|
|||||||
os.path.exists = MagicMock(return_value=positive)
|
os.path.exists = MagicMock(return_value=positive)
|
||||||
fake_spawn = _setUp_fake_spawn()
|
fake_spawn = _setUp_fake_spawn()
|
||||||
|
|
||||||
self.volumeDevice.unmount()
|
self.volumeDevice.unmount('/mnt/volume')
|
||||||
COUNT = 1
|
COUNT = 1
|
||||||
if not positive:
|
if not positive:
|
||||||
COUNT = 0
|
COUNT = 0
|
||||||
|
@ -15,19 +15,24 @@ import testtools
|
|||||||
from mock import Mock
|
from mock import Mock
|
||||||
from testtools.matchers import Equals
|
from testtools.matchers import Equals
|
||||||
from mockito import mock, when, unstub, any, verify, never
|
from mockito import mock, when, unstub, any, verify, never
|
||||||
|
from cinderclient import exceptions as cinder_exceptions
|
||||||
from trove.datastore import models as datastore_models
|
from trove.datastore import models as datastore_models
|
||||||
from trove.taskmanager import models as taskmanager_models
|
from trove.taskmanager import models as taskmanager_models
|
||||||
from trove.backup import models as backup_models
|
from trove.backup import models as backup_models
|
||||||
from trove.common import remote
|
from trove.common import remote
|
||||||
|
from trove.common.exception import GuestError
|
||||||
|
from trove.common.exception import PollTimeOut
|
||||||
from trove.common.exception import TroveError
|
from trove.common.exception import TroveError
|
||||||
from trove.common.instance import ServiceStatuses
|
from trove.common.instance import ServiceStatuses
|
||||||
from trove.extensions.mysql import models as mysql_models
|
from trove.extensions.mysql import models as mysql_models
|
||||||
from trove.instance.models import InstanceServiceStatus
|
from trove.instance.models import InstanceServiceStatus
|
||||||
|
from trove.instance.models import InstanceStatus
|
||||||
from trove.instance.models import DBInstance
|
from trove.instance.models import DBInstance
|
||||||
from trove.instance.tasks import InstanceTasks
|
from trove.instance.tasks import InstanceTasks
|
||||||
|
|
||||||
from trove.tests.unittests.util import util
|
from trove.tests.unittests.util import util
|
||||||
from trove.common import utils
|
from trove.common import utils
|
||||||
|
from trove.openstack.common import timeutils
|
||||||
from swiftclient.client import ClientException
|
from swiftclient.client import ClientException
|
||||||
from tempfile import NamedTemporaryFile
|
from tempfile import NamedTemporaryFile
|
||||||
import os
|
import os
|
||||||
@ -224,6 +229,92 @@ class FreshInstanceTasksTest(testtools.TestCase):
|
|||||||
InstanceTasks.BUILDING_ERROR_TIMEOUT_GA)
|
InstanceTasks.BUILDING_ERROR_TIMEOUT_GA)
|
||||||
|
|
||||||
|
|
||||||
|
class ResizeVolumeTest(testtools.TestCase):
|
||||||
|
def setUp(self):
|
||||||
|
super(ResizeVolumeTest, self).setUp()
|
||||||
|
utils.poll_until = Mock()
|
||||||
|
timeutils.isotime = Mock()
|
||||||
|
self.instance = Mock()
|
||||||
|
self.old_vol_size = 1
|
||||||
|
self.new_vol_size = 2
|
||||||
|
self.action = taskmanager_models.ResizeVolumeAction(self.instance,
|
||||||
|
self.old_vol_size,
|
||||||
|
self.new_vol_size)
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
super(ResizeVolumeTest, self).tearDown()
|
||||||
|
|
||||||
|
def test_resize_volume_unmount_exception(self):
|
||||||
|
self.instance.guest.unmount_volume = Mock(
|
||||||
|
side_effect=GuestError("test exception"))
|
||||||
|
self.assertRaises(GuestError,
|
||||||
|
self.action._unmount_volume,
|
||||||
|
recover_func=self.action._recover_restart)
|
||||||
|
self.assertEqual(1, self.instance.restart.call_count)
|
||||||
|
self.instance.guest.unmount_volume.side_effect = None
|
||||||
|
self.instance.reset_mock()
|
||||||
|
|
||||||
|
def test_resize_volume_detach_exception(self):
|
||||||
|
self.instance.volume_client.volumes.detach = Mock(
|
||||||
|
side_effect=cinder_exceptions.ClientException("test exception"))
|
||||||
|
self.assertRaises(cinder_exceptions.ClientException,
|
||||||
|
self.action._detach_volume,
|
||||||
|
recover_func=self.action._recover_mount_restart)
|
||||||
|
self.assertEqual(1, self.instance.guest.mount_volume.call_count)
|
||||||
|
self.assertEqual(1, self.instance.restart.call_count)
|
||||||
|
self.instance.volume_client.volumes.detach.side_effect = None
|
||||||
|
self.instance.reset_mock()
|
||||||
|
|
||||||
|
def test_resize_volume_extend_exception(self):
|
||||||
|
self.instance.volume_client.volumes.extend = Mock(
|
||||||
|
side_effect=cinder_exceptions.ClientException("test exception"))
|
||||||
|
self.assertRaises(cinder_exceptions.ClientException,
|
||||||
|
self.action._extend,
|
||||||
|
recover_func=self.action._recover_full)
|
||||||
|
attach_count = self.instance.volume_client.volumes.attach.call_count
|
||||||
|
self.assertEqual(1, attach_count)
|
||||||
|
self.assertEqual(1, self.instance.guest.mount_volume.call_count)
|
||||||
|
self.assertEqual(1, self.instance.restart.call_count)
|
||||||
|
self.instance.volume_client.volumes.extend.side_effect = None
|
||||||
|
self.instance.reset_mock()
|
||||||
|
|
||||||
|
def test_resize_volume_verify_extend_no_volume(self):
|
||||||
|
self.instance.volume_client.volumes.get = Mock(return_value=None)
|
||||||
|
self.assertRaises(cinder_exceptions.ClientException,
|
||||||
|
self.action._verify_extend)
|
||||||
|
self.instance.reset_mock()
|
||||||
|
|
||||||
|
def test_resize_volume_poll_timeout(self):
|
||||||
|
utils.poll_until = Mock(side_effect=PollTimeOut)
|
||||||
|
self.assertRaises(PollTimeOut, self.action._verify_extend)
|
||||||
|
self.assertEqual(2, self.instance.volume_client.volumes.get.call_count)
|
||||||
|
utils.poll_until.side_effect = None
|
||||||
|
self.instance.reset_mock()
|
||||||
|
|
||||||
|
def test_resize_volume_active_server_succeeds(self):
|
||||||
|
server = Mock(status=InstanceStatus.ACTIVE)
|
||||||
|
self.instance.attach_mock(server, 'server')
|
||||||
|
self.action.execute()
|
||||||
|
self.assertEqual(1, self.instance.guest.stop_db.call_count)
|
||||||
|
self.assertEqual(1, self.instance.guest.unmount_volume.call_count)
|
||||||
|
detach_count = self.instance.volume_client.volumes.detach.call_count
|
||||||
|
self.assertEqual(1, detach_count)
|
||||||
|
extend_count = self.instance.volume_client.volumes.extend.call_count
|
||||||
|
self.assertEqual(1, extend_count)
|
||||||
|
attach_count = self.instance.volume_client.volumes.attach.call_count
|
||||||
|
self.assertEqual(1, attach_count)
|
||||||
|
self.assertEqual(1, self.instance.guest.resize_fs.call_count)
|
||||||
|
self.assertEqual(1, self.instance.guest.mount_volume.call_count)
|
||||||
|
self.assertEqual(1, self.instance.restart.call_count)
|
||||||
|
self.instance.reset_mock()
|
||||||
|
|
||||||
|
def test_resize_volume_server_error_fails(self):
|
||||||
|
server = Mock(status=InstanceStatus.ERROR)
|
||||||
|
self.instance.attach_mock(server, 'server')
|
||||||
|
self.assertRaises(TroveError, self.action.execute)
|
||||||
|
self.instance.reset_mock()
|
||||||
|
|
||||||
|
|
||||||
class BackupTasksTest(testtools.TestCase):
|
class BackupTasksTest(testtools.TestCase):
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
super(BackupTasksTest, self).setUp()
|
super(BackupTasksTest, self).setUp()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user