# Copyright 2011 Quanta Research Cambridge, Inc. # # 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. """Defines various sub-classes of the `StressTestCase` and `PendingServerAction` class. Sub-classes of StressTestCase implement various API calls on the Nova cluster having to do with Server Actions. Each sub-class will have a corresponding PendingServerAction. These pending actions veriy that the API call was successful or not.""" # system imports import random import time # local imports import test_case import pending_action from tempest.exceptions import Duplicate from utils.util import * class TestRebootVM(test_case.StressTestCase): """Reboot a server""" def run(self, manager, state, *pargs, **kwargs): """ Send an HTTP POST request to the nova cluster to reboot a random server. Update state of object in `state` variable to indicate that it is rebooting. `manager` : Manager object `state` : `State` object describing our view of state of cluster `pargs` : positional arguments `kwargs` : keyword arguments, which include: `timeout` : how long to wait before issuing Exception `type` : reboot type [SOFT or HARD] (default is SOFT) """ vms = state.get_instances() active_vms = [v for k, v in vms.iteritems() if v and v[1] == 'ACTIVE'] # no active vms, so return null if not active_vms: self._logger.info('no ACTIVE instances to reboot') return _reboot_arg = kwargs.get('type', 'SOFT') # select active vm to reboot and then send request to nova controller target = random.choice(active_vms) reboot_target = target[0] # It seems that doing a reboot when in reboot is an error. try: response, body = manager.servers_client.reboot( reboot_target['id'], _reboot_arg) except Duplicate: return if (response.status != 202): self._logger.error("response: %s" % response) raise Exception if _reboot_arg == 'SOFT': reboot_state = 'REBOOT' else: reboot_state = 'HARD_REBOOT' self._logger.info('waiting for machine %s to change to %s' % (reboot_target['id'], reboot_state)) return VerifyRebootVM(manager, state, reboot_target, reboot_state=reboot_state) class VerifyRebootVM(pending_action.PendingServerAction): """Class to verify that the reboot completed.""" States = enum('REBOOT_CHECK', 'ACTIVE_CHECK') def __init__(self, manager, state, target_server, reboot_state=None, ip_addr=None): super(VerifyRebootVM, self).__init__(manager, state, target_server) self._reboot_state = reboot_state self._retry_state = self.States.REBOOT_CHECK def retry(self): """ Check to see that the server of interest has actually rebooted. Update state to indicate that server is running again. """ # don't run reboot verification if target machine has been # deleted or is going to be deleted target_id = self._target['id'] if (self._target['id'] not in self._state.get_instances().keys() or self._state.get_instances()[target_id][1] == 'TERMINATING'): self._logger.debug('machine %s is deleted or TERMINATING' % self._target['id']) return True reboot_state = self._reboot_state if self._retry_state == self.States.REBOOT_CHECK: server_state = self._check_for_status(reboot_state) if server_state == reboot_state: self._logger.info('machine %s ACTIVE -> %s' % (self._target['id'], reboot_state)) self._state.set_instance_state(self._target['id'], (self._target, reboot_state)) self._retry_state = self.States.ACTIVE_CHECK elif server_state == 'ACTIVE': # machine must have gone ACTIVE -> REBOOT ->ACTIVE self._retry_state = self.States.ACTIVE_CHECK elif self._retry_state == self.States.ACTIVE_CHECK: if not self._check_for_status('ACTIVE'): return False target = self._target self._logger.info('machine %s %s -> ACTIVE [%.1f secs elapsed]' % (target['id'], reboot_state, self.elapsed())) self._state.set_instance_state(target['id'], (target, 'ACTIVE')) return True # This code needs to be tested against a cluster that supports resize. #class TestResizeVM(test_case.StressTestCase): # """Resize a server (change flavors)""" # # def run(self, manager, state, *pargs, **kwargs): # """ # Send an HTTP POST request to the nova cluster to resize a random # server. Update `state` to indicate server is rebooting. # # `manager` : Manager object. # `state` : `State` object describing our view of state of cluster # `pargs` : positional arguments # `kwargs` : keyword arguments, which include: # `timeout` : how long to wait before issuing Exception # """ # # vms = state.get_instances() # active_vms = [v for k, v in vms.iteritems() if v and v[1] == 'ACTIVE'] # # no active vms, so return null # if not active_vms: # self._logger.debug('no ACTIVE instances to resize') # return # # target = random.choice(active_vms) # resize_target = target[0] # print resize_target # # _timeout = kwargs.get('timeout', 600) # # # determine current flavor type, and resize to a different type # # m1.tiny -> m1.small, m1.small -> m1.tiny # curr_size = int(resize_target['flavor']['id']) # if curr_size == 1: # new_size = 2 # else: # new_size = 1 # flavor_type = { 'flavorRef': new_size } # resize to m1.small # # post_body = json.dumps({'resize' : flavor_type}) # url = '/servers/%s/action' % resize_target['id'] # (response, body) = manager.request('POST', # url, # body=post_body) # # if (response.status != 202): # self._logger.error("response: %s" % response) # raise Exception # # state_name = check_for_status(manager, resize_target, 'RESIZE') # # if state_name == 'RESIZE': # self._logger.info('machine %s: ACTIVE -> RESIZE' % # resize_target['id']) # state.set_instance_state(resize_target['id'], # (resize_target, 'RESIZE')) # # return VerifyResizeVM(manager, # state, # resize_target, # state_name=state_name, # timeout=_timeout) # #class VerifyResizeVM(pending_action.PendingServerAction): # """Verify that resizing of a VM was successful""" # States = enum('VERIFY_RESIZE_CHECK', 'ACTIVE_CHECK') # # def __init__(self, manager, state, created_server, # state_name=None, # timeout=300): # super(VerifyResizeVM, self).__init__(manager, # state, # created_server, # timeout=timeout) # self._retry_state = self.States.VERIFY_RESIZE_CHECK # self._state_name = state_name # # def retry(self): # """ # Check to see that the server was actually resized. And change `state` # of server to running again. # """ # # don't run resize if target machine has been deleted # # or is going to be deleted # if (self._target['id'] not in self._state.get_instances().keys() or # self._state.get_instances()[self._target['id']][1] == # 'TERMINATING'): # self._logger.debug('machine %s is deleted or TERMINATING' % # self._target['id']) # return True # # if self._retry_state == self.States.VERIFY_RESIZE_CHECK: # if self._check_for_status('VERIFY_RESIZE') == 'VERIFY_RESIZE': # # now issue command to CONFIRM RESIZE # post_body = json.dumps({'confirmResize' : null}) # url = '/servers/%s/action' % self._target['id'] # (response, body) = manager.request('POST', # url, # body=post_body) # if (response.status != 204): # self._logger.error("response: %s" % response) # raise Exception # # self._logger.info( # 'CONFIRMING RESIZE of machine %s [%.1f secs elapsed]' % # (self._target['id'], self.elapsed()) # ) # state.set_instance_state(self._target['id'], # (self._target, 'CONFIRM_RESIZE')) # # # change states # self._retry_state = self.States.ACTIVE_CHECK # # return False # # elif self._retry_state == self.States.ACTIVE_CHECK: # if not self._check_manager("ACTIVE"): # return False # else: # server = self._manager.get_server(self._target['id']) # # # Find private IP of server? # try: # (_, network) = server['addresses'].popitem() # ip = network[0]['addr'] # except KeyError: # self._logger.error( # 'could not get ip address for machine %s' % # self._target['id'] # ) # raise Exception # # self._logger.info( # 'machine %s: VERIFY_RESIZE -> ACTIVE [%.1f sec elapsed]' % # (self._target['id'], self.elapsed()) # ) # self._state.set_instance_state(self._target['id'], # (self._target, 'ACTIVE')) # # return True # # else: # # should never get here # self._logger.error('Unexpected state') # raise Exception