blob: 5fba97b6f30b4f2621f6d7d9821cffa5f016acae [file] [log] [blame]
#
#
# Copyright (C) 2006, 2007, 2008, 2009, 2010, 2011, 2012, 2013 Google Inc.
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are
# met:
#
# 1. Redistributions of source code must retain the above copyright notice,
# this list of conditions and the following disclaimer.
#
# 2. Redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
# IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
# TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
# EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
# PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
# PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
# LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
# NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
"""Functions for running helper virtual machines to perform tasks on instances.
"""
import contextlib
from ganeti import constants
from ganeti import errors
from ganeti.utils import retry
from ganeti.cmdlib.common import IsInstanceRunning, DetermineImageSize
from ganeti.cmdlib.instance_storage import StartInstanceDisks, \
TemporaryDisk, ImageDisks
@contextlib.contextmanager
def HelperVM(lu, instance, vm_image, startup_timeout, vm_timeout,
log_prefix=None, feedback_fn=None):
"""Runs a given helper VM for a given instance.
@type lu: L{LogicalUnit}
@param lu: the lu on whose behalf we execute
@type instance: L{objects.Instance}
@param instance: the instance definition
@type vm_image: string
@param vm_image: the name of the helper VM image to dump on a temporary disk
@type startup_timeout: int
@param startup_timeout: how long to wait for the helper VM to start up
@type vm_timeout: int
@param vm_timeout: how long to wait for the helper VM to finish its work
@type log_prefix: string
@param log_prefix: a prefix for all log messages
@type feedback_fn: function
@param feedback_fn: Function used to log progress
"""
if log_prefix:
add_prefix = lambda msg: "%s: %s" % (log_prefix, msg)
else:
add_prefix = lambda msg: msg
if feedback_fn is not None:
log_feedback = lambda msg: feedback_fn(add_prefix(msg))
else:
log_feedback = lambda _: None
try:
disk_size = DetermineImageSize(lu, vm_image, instance.primary_node)
except errors.OpExecError, err:
raise errors.OpExecError("Could not create temporary disk: %s", err)
with TemporaryDisk(lu,
instance,
[(constants.DT_PLAIN, constants.DISK_RDWR, disk_size)],
log_feedback):
log_feedback("Activating helper VM's temporary disks")
StartInstanceDisks(lu, instance, False)
log_feedback("Imaging temporary disks with image %s" % (vm_image, ))
ImageDisks(lu, instance, vm_image)
log_feedback("Starting helper VM")
result = lu.rpc.call_instance_start(instance.primary_node,
(instance, [], []),
False, lu.op.reason)
result.Raise(add_prefix("Could not start helper VM with image %s" %
(vm_image, )))
# First wait for the instance to start up
running_check = lambda: IsInstanceRunning(lu, instance, prereq=False)
instance_up = retry.SimpleRetry(True, running_check, 5.0,
startup_timeout)
if not instance_up:
raise errors.OpExecError(add_prefix("Could not boot instance using"
" image %s" % (vm_image, )))
log_feedback("Helper VM is up")
def cleanup():
log_feedback("Waiting for helper VM to finish")
# Then for it to be finished, detected by its shutdown
instance_up = retry.SimpleRetry(False, running_check, 20.0, vm_timeout)
if instance_up:
lu.LogWarning(add_prefix("Helper VM has not finished within the"
" timeout; shutting it down forcibly"))
return \
lu.rpc.call_instance_shutdown(instance.primary_node,
instance,
constants.DEFAULT_SHUTDOWN_TIMEOUT,
lu.op.reason)
else:
return None
# Run the inner block and handle possible errors
try:
yield
except Exception:
# if the cleanup failed for some reason, log it and just re-raise
result = cleanup()
if result:
result.Warn(add_prefix("Could not shut down helper VM with image"
" %s within timeout" % (vm_image, )))
log_feedback("Error running helper VM with image %s" %
(vm_image, ))
raise
else:
result = cleanup()
# if the cleanup failed for some reason, throw an exception
if result:
result.Raise(add_prefix("Could not shut down helper VM with image %s"
" within timeout" % (vm_image, )))
raise errors.OpExecError("Error running helper VM with image %s" %
(vm_image, ))
log_feedback("Helper VM execution completed")
def RunWithHelperVM(lu, instance, vm_image, startup_timeout, vm_timeout,
log_prefix=None, feedback_fn=None):
"""Runs a given helper VM for a given instance.
@type lu: L{LogicalUnit}
@param lu: the lu on whose behalf we execute
@type instance: L{objects.Instance}
@param instance: the instance definition
@type vm_image: string
@param vm_image: the name of the helper VM image to dump on a temporary disk
@type startup_timeout: int
@param startup_timeout: how long to wait for the helper VM to start up
@type vm_timeout: int
@param vm_timeout: how long to wait for the helper VM to finish its work
@type log_prefix: string
@param log_prefix: a prefix for all log messages
@type feedback_fn: function
@param feedback_fn: Function used to log progress
"""
with HelperVM(lu, instance, vm_image, startup_timeout, vm_timeout,
log_prefix=log_prefix, feedback_fn=feedback_fn):
pass