compute-hyperv/compute_hyperv/nova/cluster/livemigrationops.py

125 lines
5.3 KiB
Python

# Copyright 2016 Cloudbase Solutions Srl
# All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
"""Management class for cluster live migration VM operations."""
from nova.compute import vm_states
from nova import exception
from os_win import constants as os_win_const
from os_win import utilsfactory
from oslo_log import log as logging
from oslo_utils import excutils
from compute_hyperv.i18n import _
import compute_hyperv.nova.conf
from compute_hyperv.nova import livemigrationops
CONF = compute_hyperv.nova.conf.CONF
LOG = logging.getLogger(__name__)
class ClusterLiveMigrationOps(livemigrationops.LiveMigrationOps):
def __init__(self):
super(ClusterLiveMigrationOps, self).__init__()
self._clustutils = utilsfactory.get_clusterutils()
def is_instance_clustered(self, instance_name):
return self._clustutils.vm_exists(instance_name)
def live_migration(self, context, instance_ref, dest, post_method,
recover_method, block_migration=False,
migrate_data=None):
LOG.debug("live_migration called.", instance=instance_ref)
instance_name = instance_ref.name
clustered = self.is_instance_clustered(instance_name)
node_names = [node.upper() for node in
self._clustutils.get_cluster_node_names()]
if dest.upper() not in node_names or not clustered:
# destination is not in same cluster or instance not clustered.
# do a normal live migration.
if clustered:
# remove VM from cluster before proceding to a normal live
# migration.
self._clustutils.delete(instance_name)
super(ClusterLiveMigrationOps, self).live_migration(
context, instance_ref, dest, post_method, recover_method,
block_migration, migrate_data)
return
elif self._clustutils.get_vm_host(
instance_name).upper() == dest.upper():
# VM is already migrated. Do nothing.
# this can happen when the VM has been failovered.
return
# destination is in the same cluster.
# perform a clustered live migration.
try:
self._clustutils.live_migrate_vm(
instance_name,
dest,
CONF.hyperv.instance_live_migration_timeout)
except Exception:
with excutils.save_and_reraise_exception():
self._check_failed_instance_migration(
instance_ref,
expected_state=os_win_const.CLUSTER_GROUP_ONLINE)
LOG.debug("Calling live migration recover_method "
"for instance.", instance=instance_ref)
recover_method(context, instance_ref, dest, block_migration,
migrate_data)
LOG.debug("Calling live migration post_method for instance.",
instance=instance_ref)
post_method(context, instance_ref, dest,
block_migration, migrate_data)
def _check_failed_instance_migration(self, instance, expected_state):
# After a failed migration, we expect the instance to be on the
# source node, having its initial state and not have any queued
# migrations. Otherwise, we treat it as a critical error and set
# it to 'error' state to avoid inconsistencies.
state_info = self._clustutils.get_cluster_group_state_info(
instance.name)
node_name = self._clustutils.get_node_name()
if (state_info['owner_node'].lower() != node_name.lower()
or state_info['state'] != expected_state
or state_info['migration_queued']):
instance.vm_state = vm_states.ERROR
instance.save()
raise exception.InstanceInvalidState(
_("Instance %(instance_name)s reached an inconsistent state "
"after a failed migration attempt. Setting the instance to "
"'error' state. Instance state info: %(state_info)s.") %
dict(instance_name=instance.name,
state_info=state_info))
def pre_live_migration(self, context, instance, block_device_info,
network_info):
if self.is_instance_clustered(instance.name):
self._volumeops.connect_volumes(block_device_info)
else:
super(ClusterLiveMigrationOps, self).pre_live_migration(
context, instance, block_device_info, network_info)
def post_live_migration(self, context, instance, block_device_info,
migrate_data):
if not self.is_instance_clustered(instance.name):
super(ClusterLiveMigrationOps, self).post_live_migration(
context, instance, block_device_info, migrate_data)