1704 lines
		
	
	
		
			70 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			1704 lines
		
	
	
		
			70 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # vim: tabstop=4 shiftwidth=4 softtabstop=4
 | |
| 
 | |
| # Copyright 2010 United States Government as represented by the
 | |
| # Administrator of the National Aeronautics and Space Administration.
 | |
| # Copyright 2011 Piston Cloud Computing, Inc.
 | |
| # 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.
 | |
| """
 | |
| Tests For Compute
 | |
| """
 | |
| 
 | |
| from copy import copy
 | |
| import mox
 | |
| 
 | |
| from nova import compute
 | |
| from nova import context
 | |
| from nova import db
 | |
| from nova import exception
 | |
| from nova import flags
 | |
| from nova import log as logging
 | |
| from nova.scheduler import driver as scheduler_driver
 | |
| from nova import rpc
 | |
| from nova import test
 | |
| from nova import utils
 | |
| import nova
 | |
| 
 | |
| from nova.compute import instance_types
 | |
| from nova.compute import manager as compute_manager
 | |
| from nova.compute import power_state
 | |
| from nova.compute import task_states
 | |
| from nova.compute import vm_states
 | |
| from nova.db.sqlalchemy import models
 | |
| from nova.image import fake as fake_image
 | |
| from nova.notifier import test_notifier
 | |
| from nova.tests import fake_network
 | |
| from nova.network.quantum import client as quantum_client
 | |
| 
 | |
| 
 | |
| LOG = logging.getLogger('nova.tests.compute')
 | |
| FLAGS = flags.FLAGS
 | |
| flags.DECLARE('stub_network', 'nova.compute.manager')
 | |
| flags.DECLARE('live_migration_retry_count', 'nova.compute.manager')
 | |
| 
 | |
| 
 | |
| class FakeTime(object):
 | |
|     def __init__(self):
 | |
|         self.counter = 0
 | |
| 
 | |
|     def sleep(self, t):
 | |
|         self.counter += t
 | |
| 
 | |
| 
 | |
| orig_rpc_call = rpc.call
 | |
| orig_rpc_cast = rpc.cast
 | |
| 
 | |
| 
 | |
| def rpc_call_wrapper(context, topic, msg, do_cast=True):
 | |
|     """Stub out the scheduler creating the instance entry"""
 | |
|     if topic == FLAGS.scheduler_topic and \
 | |
|             msg['method'] == 'run_instance':
 | |
|         request_spec = msg['args']['request_spec']
 | |
|         scheduler = scheduler_driver.Scheduler
 | |
|         num_instances = request_spec.get('num_instances', 1)
 | |
|         instances = []
 | |
|         for x in xrange(num_instances):
 | |
|             instance = scheduler().create_instance_db_entry(
 | |
|                     context, request_spec)
 | |
|             encoded = scheduler_driver.encode_instance(instance)
 | |
|             instances.append(encoded)
 | |
|         return instances
 | |
|     else:
 | |
|         if do_cast:
 | |
|             orig_rpc_cast(context, topic, msg)
 | |
|         else:
 | |
|             return orig_rpc_call(context, topic, msg)
 | |
| 
 | |
| 
 | |
| def rpc_cast_wrapper(context, topic, msg):
 | |
|     """Stub out the scheduler creating the instance entry in
 | |
|     the reservation_id case.
 | |
|     """
 | |
|     rpc_call_wrapper(context, topic, msg, do_cast=True)
 | |
| 
 | |
| 
 | |
| def nop_report_driver_status(self):
 | |
|     pass
 | |
| 
 | |
| 
 | |
| class ComputeTestCase(test.TestCase):
 | |
|     """Test case for compute"""
 | |
|     def setUp(self):
 | |
|         super(ComputeTestCase, self).setUp()
 | |
|         self.flags(connection_type='fake',
 | |
|                    stub_network=True,
 | |
|                    notification_driver='nova.notifier.test_notifier',
 | |
|                    network_manager='nova.network.manager.FlatManager')
 | |
|         self.compute = utils.import_object(FLAGS.compute_manager)
 | |
|         self.compute_api = compute.API()
 | |
|         self.user_id = 'fake'
 | |
|         self.project_id = 'fake'
 | |
|         self.context = context.RequestContext(self.user_id, self.project_id)
 | |
|         test_notifier.NOTIFICATIONS = []
 | |
| 
 | |
|         def fake_show(meh, context, id):
 | |
|             return {'id': 1, 'min_disk': None, 'min_ram': None,
 | |
|                     'properties': {'kernel_id': 1, 'ramdisk_id': 1}}
 | |
| 
 | |
|         self.stubs.Set(fake_image._FakeImageService, 'show', fake_show)
 | |
|         self.stubs.Set(rpc, 'call', rpc_call_wrapper)
 | |
|         self.stubs.Set(rpc, 'cast', rpc_cast_wrapper)
 | |
| 
 | |
|     def _create_instance(self, params=None):
 | |
|         """Create a test instance"""
 | |
|         if not params:
 | |
|             params = {}
 | |
| 
 | |
|         inst = {}
 | |
|         inst['image_ref'] = 1
 | |
|         inst['reservation_id'] = 'r-fakeres'
 | |
|         inst['launch_time'] = '10'
 | |
|         inst['user_id'] = self.user_id
 | |
|         inst['project_id'] = self.project_id
 | |
|         type_id = instance_types.get_instance_type_by_name('m1.tiny')['id']
 | |
|         inst['instance_type_id'] = type_id
 | |
|         inst['ami_launch_index'] = 0
 | |
|         inst.update(params)
 | |
|         return db.instance_create(self.context, inst)['id']
 | |
| 
 | |
|     def _create_instance_type(self, params=None):
 | |
|         """Create a test instance"""
 | |
|         if not params:
 | |
|             params = {}
 | |
| 
 | |
|         context = self.context.elevated()
 | |
|         inst = {}
 | |
|         inst['name'] = 'm1.small'
 | |
|         inst['memory_mb'] = '1024'
 | |
|         inst['vcpus'] = '1'
 | |
|         inst['local_gb'] = '20'
 | |
|         inst['flavorid'] = '1'
 | |
|         inst['swap'] = '2048'
 | |
|         inst['rxtx_quota'] = 100
 | |
|         inst['rxtx_cap'] = 200
 | |
|         inst.update(params)
 | |
|         return db.instance_type_create(context, inst)['id']
 | |
| 
 | |
|     def _create_group(self):
 | |
|         values = {'name': 'testgroup',
 | |
|                   'description': 'testgroup',
 | |
|                   'user_id': self.user_id,
 | |
|                   'project_id': self.project_id}
 | |
|         return db.security_group_create(self.context, values)
 | |
| 
 | |
|     def test_create_instance_defaults_display_name(self):
 | |
|         """Verify that an instance cannot be created without a display_name."""
 | |
|         cases = [dict(), dict(display_name=None)]
 | |
|         for instance in cases:
 | |
|             (ref, resv_id) = self.compute_api.create(self.context,
 | |
|                 instance_types.get_default_instance_type(), None, **instance)
 | |
|             try:
 | |
|                 self.assertNotEqual(ref[0]['display_name'], None)
 | |
|             finally:
 | |
|                 db.instance_destroy(self.context, ref[0]['id'])
 | |
| 
 | |
|     def test_create_instance_associates_security_groups(self):
 | |
|         """Make sure create associates security groups"""
 | |
|         group = self._create_group()
 | |
|         (ref, resv_id) = self.compute_api.create(
 | |
|                 self.context,
 | |
|                 instance_type=instance_types.get_default_instance_type(),
 | |
|                 image_href=None,
 | |
|                 security_group=['testgroup'])
 | |
|         try:
 | |
|             self.assertEqual(len(db.security_group_get_by_instance(
 | |
|                              self.context, ref[0]['id'])), 1)
 | |
|             group = db.security_group_get(self.context, group['id'])
 | |
|             self.assert_(len(group.instances) == 1)
 | |
|         finally:
 | |
|             db.security_group_destroy(self.context, group['id'])
 | |
|             db.instance_destroy(self.context, ref[0]['id'])
 | |
| 
 | |
|     def test_create_instance_with_invalid_security_group_raises(self):
 | |
|         instance_type = instance_types.get_default_instance_type()
 | |
| 
 | |
|         pre_build_len = len(db.instance_get_all(context.get_admin_context()))
 | |
|         self.assertRaises(exception.SecurityGroupNotFoundForProject,
 | |
|                           self.compute_api.create,
 | |
|                           self.context,
 | |
|                           instance_type=instance_type,
 | |
|                           image_href=None,
 | |
|                           security_group=['this_is_a_fake_sec_group'])
 | |
|         self.assertEqual(pre_build_len,
 | |
|                          len(db.instance_get_all(context.get_admin_context())))
 | |
| 
 | |
|     def test_create_instance_with_img_ref_associates_config_drive(self):
 | |
|         """Make sure create associates a config drive."""
 | |
| 
 | |
|         instance_id = self._create_instance(params={'config_drive': '1234', })
 | |
| 
 | |
|         try:
 | |
|             self.compute.run_instance(self.context, instance_id)
 | |
|             instances = db.instance_get_all(context.get_admin_context())
 | |
|             instance = instances[0]
 | |
| 
 | |
|             self.assertTrue(instance.config_drive)
 | |
|         finally:
 | |
|             db.instance_destroy(self.context, instance_id)
 | |
| 
 | |
|     def test_create_instance_associates_config_drive(self):
 | |
|         """Make sure create associates a config drive."""
 | |
| 
 | |
|         instance_id = self._create_instance(params={'config_drive': True, })
 | |
| 
 | |
|         try:
 | |
|             self.compute.run_instance(self.context, instance_id)
 | |
|             instances = db.instance_get_all(context.get_admin_context())
 | |
|             instance = instances[0]
 | |
| 
 | |
|             self.assertTrue(instance.config_drive)
 | |
|         finally:
 | |
|             db.instance_destroy(self.context, instance_id)
 | |
| 
 | |
|     def test_default_hostname_generator(self):
 | |
|         cases = [(None, 'server-1'), ('Hello, Server!', 'hello-server'),
 | |
|                  ('<}\x1fh\x10e\x08l\x02l\x05o\x12!{>', 'hello'),
 | |
|                  ('hello_server', 'hello-server')]
 | |
|         for display_name, hostname in cases:
 | |
|             (ref, resv_id) = self.compute_api.create(self.context,
 | |
|                 instance_types.get_default_instance_type(), None,
 | |
|                 display_name=display_name)
 | |
|             try:
 | |
|                 self.assertEqual(ref[0]['hostname'], hostname)
 | |
|             finally:
 | |
|                 db.instance_destroy(self.context, ref[0]['id'])
 | |
| 
 | |
|     def test_destroy_instance_disassociates_security_groups(self):
 | |
|         """Make sure destroying disassociates security groups"""
 | |
|         group = self._create_group()
 | |
| 
 | |
|         (ref, resv_id) = self.compute_api.create(
 | |
|                 self.context,
 | |
|                 instance_type=instance_types.get_default_instance_type(),
 | |
|                 image_href=None,
 | |
|                 security_group=['testgroup'])
 | |
|         try:
 | |
|             db.instance_destroy(self.context, ref[0]['id'])
 | |
|             group = db.security_group_get(self.context, group['id'])
 | |
|             self.assert_(len(group.instances) == 0)
 | |
|         finally:
 | |
|             db.security_group_destroy(self.context, group['id'])
 | |
| 
 | |
|     def test_destroy_security_group_disassociates_instances(self):
 | |
|         """Make sure destroying security groups disassociates instances"""
 | |
|         group = self._create_group()
 | |
| 
 | |
|         (ref, resv_id) = self.compute_api.create(
 | |
|                 self.context,
 | |
|                 instance_type=instance_types.get_default_instance_type(),
 | |
|                 image_href=None,
 | |
|                 security_group=['testgroup'])
 | |
| 
 | |
|         try:
 | |
|             db.security_group_destroy(self.context, group['id'])
 | |
|             group = db.security_group_get(context.get_admin_context(
 | |
|                                           read_deleted=True), group['id'])
 | |
|             self.assert_(len(group.instances) == 0)
 | |
|         finally:
 | |
|             db.instance_destroy(self.context, ref[0]['id'])
 | |
| 
 | |
|     def test_run_terminate(self):
 | |
|         """Make sure it is possible to  run and terminate instance"""
 | |
|         instance_id = self._create_instance()
 | |
| 
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
| 
 | |
|         instances = db.instance_get_all(context.get_admin_context())
 | |
|         LOG.info(_("Running instances: %s"), instances)
 | |
|         self.assertEqual(len(instances), 1)
 | |
| 
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|         instances = db.instance_get_all(context.get_admin_context())
 | |
|         LOG.info(_("After terminating instances: %s"), instances)
 | |
|         self.assertEqual(len(instances), 0)
 | |
| 
 | |
|     def test_run_terminate_timestamps(self):
 | |
|         """Make sure timestamps are set for launched and destroyed"""
 | |
|         instance_id = self._create_instance()
 | |
|         instance_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertEqual(instance_ref['launched_at'], None)
 | |
|         self.assertEqual(instance_ref['deleted_at'], None)
 | |
|         launch = utils.utcnow()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         instance_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assert_(instance_ref['launched_at'] > launch)
 | |
|         self.assertEqual(instance_ref['deleted_at'], None)
 | |
|         terminate = utils.utcnow()
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
|         self.context = self.context.elevated(True)
 | |
|         instance_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assert_(instance_ref['launched_at'] < terminate)
 | |
|         self.assert_(instance_ref['deleted_at'] > terminate)
 | |
| 
 | |
|     def test_stop(self):
 | |
|         """Ensure instance can be stopped"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         self.compute.stop_instance(self.context, instance_id)
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_start(self):
 | |
|         """Ensure instance can be started"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         self.compute.stop_instance(self.context, instance_id)
 | |
|         self.compute.start_instance(self.context, instance_id)
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_pause(self):
 | |
|         """Ensure instance can be paused"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         self.compute.pause_instance(self.context, instance_id)
 | |
|         self.compute.unpause_instance(self.context, instance_id)
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_suspend(self):
 | |
|         """ensure instance can be suspended"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         self.compute.suspend_instance(self.context, instance_id)
 | |
|         self.compute.resume_instance(self.context, instance_id)
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_soft_reboot_api(self):
 | |
|         """Ensure instance can be soft rebooted"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
| 
 | |
|         inst_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertEqual(inst_ref['task_state'], None)
 | |
| 
 | |
|         reboot_type = "SOFT"
 | |
|         self.compute_api.reboot(self.context, instance_id, reboot_type)
 | |
| 
 | |
|         inst_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertEqual(inst_ref['task_state'], task_states.REBOOTING)
 | |
| 
 | |
|         db.instance_destroy(self.context, instance_id)
 | |
| 
 | |
|     def test_soft_reboot(self):
 | |
|         """Ensure instance can be soft rebooted"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         db.instance_update(self.context, instance_id,
 | |
|                            {'task_state': task_states.REBOOTING})
 | |
| 
 | |
|         reboot_type = "SOFT"
 | |
|         self.compute.reboot_instance(self.context, instance_id, reboot_type)
 | |
| 
 | |
|         inst_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertEqual(inst_ref['power_state'], power_state.RUNNING)
 | |
|         self.assertEqual(inst_ref['task_state'], None)
 | |
| 
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_hard_reboot_api(self):
 | |
|         """Ensure instance can be hard rebooted"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
| 
 | |
|         inst_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertEqual(inst_ref['task_state'], None)
 | |
| 
 | |
|         reboot_type = "HARD"
 | |
|         self.compute_api.reboot(self.context, instance_id, reboot_type)
 | |
| 
 | |
|         inst_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertEqual(inst_ref['task_state'], task_states.REBOOTING_HARD)
 | |
| 
 | |
|         db.instance_destroy(self.context, instance_id)
 | |
| 
 | |
|     def test_hard_reboot(self):
 | |
|         """Ensure instance can be hard rebooted"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         db.instance_update(self.context, instance_id,
 | |
|                            {'task_state': task_states.REBOOTING_HARD})
 | |
| 
 | |
|         reboot_type = "HARD"
 | |
|         self.compute.reboot_instance(self.context, instance_id, reboot_type)
 | |
| 
 | |
|         inst_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertEqual(inst_ref['power_state'], power_state.RUNNING)
 | |
|         self.assertEqual(inst_ref['task_state'], None)
 | |
| 
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_set_admin_password(self):
 | |
|         """Ensure instance can have its admin password set"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         db.instance_update(self.context, instance_id,
 | |
|                            {'task_state': task_states.UPDATING_PASSWORD})
 | |
| 
 | |
|         inst_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertEqual(inst_ref['vm_state'], vm_states.ACTIVE)
 | |
|         self.assertEqual(inst_ref['task_state'], task_states.UPDATING_PASSWORD)
 | |
| 
 | |
|         self.compute.set_admin_password(self.context, instance_id)
 | |
| 
 | |
|         inst_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertEqual(inst_ref['vm_state'], vm_states.ACTIVE)
 | |
|         self.assertEqual(inst_ref['task_state'], None)
 | |
| 
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_set_admin_password_api(self):
 | |
|         """Ensure instance can have its admin password set"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
| 
 | |
|         inst_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertEqual(inst_ref['vm_state'], vm_states.ACTIVE)
 | |
|         self.assertEqual(inst_ref['task_state'], None)
 | |
| 
 | |
|         self.compute_api.set_admin_password(self.context, instance_id)
 | |
| 
 | |
|         inst_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertEqual(inst_ref['vm_state'], vm_states.ACTIVE)
 | |
|         self.assertEqual(inst_ref['task_state'], task_states.UPDATING_PASSWORD)
 | |
| 
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_inject_file(self):
 | |
|         """Ensure we can write a file to an instance"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         self.compute.inject_file(self.context, instance_id, "/tmp/test",
 | |
|                 "File Contents")
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_agent_update(self):
 | |
|         """Ensure instance can have its agent updated"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         self.compute.agent_update(self.context, instance_id,
 | |
|                 'http://127.0.0.1/agent', '00112233445566778899aabbccddeeff')
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_snapshot(self):
 | |
|         """Ensure instance can be snapshotted"""
 | |
|         instance_id = self._create_instance()
 | |
|         name = "myfakesnapshot"
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         self.compute.snapshot_instance(self.context, instance_id, name)
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_snapshot_conflict_backup(self):
 | |
|         """Can't backup an instance which is already being backed up."""
 | |
|         instance_id = self._create_instance()
 | |
|         instance_values = {'task_state': task_states.IMAGE_BACKUP}
 | |
|         db.instance_update(self.context, instance_id, instance_values)
 | |
| 
 | |
|         self.assertRaises(exception.InstanceBackingUp,
 | |
|                           self.compute_api.backup,
 | |
|                           self.context,
 | |
|                           instance_id,
 | |
|                           None,
 | |
|                           None,
 | |
|                           None)
 | |
| 
 | |
|         db.instance_destroy(self.context, instance_id)
 | |
| 
 | |
|     def test_snapshot_conflict_snapshot(self):
 | |
|         """Can't snapshot an instance which is already being snapshotted."""
 | |
|         instance_id = self._create_instance()
 | |
|         instance_values = {'task_state': task_states.IMAGE_SNAPSHOT}
 | |
|         db.instance_update(self.context, instance_id, instance_values)
 | |
| 
 | |
|         self.assertRaises(exception.InstanceSnapshotting,
 | |
|                           self.compute_api.snapshot,
 | |
|                           self.context,
 | |
|                           instance_id,
 | |
|                           None)
 | |
| 
 | |
|         db.instance_destroy(self.context, instance_id)
 | |
| 
 | |
|     def test_console_output(self):
 | |
|         """Make sure we can get console output from instance"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
| 
 | |
|         console = self.compute.get_console_output(self.context,
 | |
|                                                         instance_id)
 | |
|         self.assert_(console)
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_ajax_console(self):
 | |
|         """Make sure we can get console output from instance"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
| 
 | |
|         console = self.compute.get_ajax_console(self.context,
 | |
|                                                 instance_id)
 | |
|         self.assert_(set(['token', 'host', 'port']).issubset(console.keys()))
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_vnc_console(self):
 | |
|         """Make sure we can a vnc console for an instance."""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
| 
 | |
|         console = self.compute.get_vnc_console(self.context,
 | |
|                                                instance_id)
 | |
|         self.assert_(console)
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_add_fixed_ip_usage_notification(self):
 | |
|         def dummy(*args, **kwargs):
 | |
|             pass
 | |
| 
 | |
|         self.stubs.Set(nova.network.API, 'add_fixed_ip_to_instance',
 | |
|                        dummy)
 | |
|         self.stubs.Set(nova.compute.manager.ComputeManager,
 | |
|                        'inject_network_info', dummy)
 | |
|         self.stubs.Set(nova.compute.manager.ComputeManager,
 | |
|                        'reset_network', dummy)
 | |
| 
 | |
|         instance_id = self._create_instance()
 | |
| 
 | |
|         self.assertEquals(len(test_notifier.NOTIFICATIONS), 0)
 | |
|         self.compute.add_fixed_ip_to_instance(self.context,
 | |
|                                               instance_id,
 | |
|                                               1)
 | |
| 
 | |
|         self.assertEquals(len(test_notifier.NOTIFICATIONS), 1)
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_remove_fixed_ip_usage_notification(self):
 | |
|         def dummy(*args, **kwargs):
 | |
|             pass
 | |
| 
 | |
|         self.stubs.Set(nova.network.API, 'remove_fixed_ip_from_instance',
 | |
|                        dummy)
 | |
|         self.stubs.Set(nova.compute.manager.ComputeManager,
 | |
|                        'inject_network_info', dummy)
 | |
|         self.stubs.Set(nova.compute.manager.ComputeManager,
 | |
|                        'reset_network', dummy)
 | |
| 
 | |
|         instance_id = self._create_instance()
 | |
| 
 | |
|         self.assertEquals(len(test_notifier.NOTIFICATIONS), 0)
 | |
|         self.compute.remove_fixed_ip_from_instance(self.context,
 | |
|                                               instance_id,
 | |
|                                               1)
 | |
| 
 | |
|         self.assertEquals(len(test_notifier.NOTIFICATIONS), 1)
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_run_instance_usage_notification(self):
 | |
|         """Ensure run instance generates apropriate usage notification"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         self.assertEquals(len(test_notifier.NOTIFICATIONS), 1)
 | |
|         inst_ref = db.instance_get(self.context, instance_id)
 | |
|         msg = test_notifier.NOTIFICATIONS[0]
 | |
|         self.assertEquals(msg['priority'], 'INFO')
 | |
|         self.assertEquals(msg['event_type'], 'compute.instance.create')
 | |
|         payload = msg['payload']
 | |
|         self.assertEquals(payload['tenant_id'], self.project_id)
 | |
|         self.assertEquals(payload['user_id'], self.user_id)
 | |
|         self.assertEquals(payload['instance_id'], inst_ref.uuid)
 | |
|         self.assertEquals(payload['instance_type'], 'm1.tiny')
 | |
|         type_id = instance_types.get_instance_type_by_name('m1.tiny')['id']
 | |
|         self.assertEquals(str(payload['instance_type_id']), str(type_id))
 | |
|         self.assertTrue('display_name' in payload)
 | |
|         self.assertTrue('created_at' in payload)
 | |
|         self.assertTrue('launched_at' in payload)
 | |
|         self.assertEquals(payload['image_ref'], '1')
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_terminate_usage_notification(self):
 | |
|         """Ensure terminate_instance generates apropriate usage notification"""
 | |
|         instance_id = self._create_instance()
 | |
|         inst_ref = db.instance_get(self.context, instance_id)
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         test_notifier.NOTIFICATIONS = []
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|         self.assertEquals(len(test_notifier.NOTIFICATIONS), 2)
 | |
|         msg = test_notifier.NOTIFICATIONS[0]
 | |
|         self.assertEquals(msg['priority'], 'INFO')
 | |
|         self.assertEquals(msg['event_type'], 'compute.instance.exists')
 | |
| 
 | |
|         msg = test_notifier.NOTIFICATIONS[1]
 | |
|         self.assertEquals(msg['priority'], 'INFO')
 | |
|         self.assertEquals(msg['event_type'], 'compute.instance.delete')
 | |
|         payload = msg['payload']
 | |
|         self.assertEquals(payload['tenant_id'], self.project_id)
 | |
|         self.assertEquals(payload['user_id'], self.user_id)
 | |
|         self.assertEquals(payload['instance_id'], inst_ref.uuid)
 | |
|         self.assertEquals(payload['instance_type'], 'm1.tiny')
 | |
|         type_id = instance_types.get_instance_type_by_name('m1.tiny')['id']
 | |
|         self.assertEquals(str(payload['instance_type_id']), str(type_id))
 | |
|         self.assertTrue('display_name' in payload)
 | |
|         self.assertTrue('created_at' in payload)
 | |
|         self.assertTrue('launched_at' in payload)
 | |
|         self.assertEquals(payload['image_ref'], '1')
 | |
| 
 | |
|     def test_run_instance_existing(self):
 | |
|         """Ensure failure when running an instance that already exists"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         self.assertRaises(exception.Error,
 | |
|                           self.compute.run_instance,
 | |
|                           self.context,
 | |
|                           instance_id)
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_instance_set_to_error_on_uncaught_exception(self):
 | |
|         """Test that instance is set to error state when exception is raised"""
 | |
|         instance_id = self._create_instance()
 | |
| 
 | |
|         self.mox.StubOutWithMock(self.compute.network_api,
 | |
|                                  "allocate_for_instance")
 | |
|         self.compute.network_api.allocate_for_instance(mox.IgnoreArg(),
 | |
|                                                        mox.IgnoreArg(),
 | |
|                                                        requested_networks=None,
 | |
|                                                        vpn=False).\
 | |
|             AndRaise(quantum_client.QuantumServerException())
 | |
| 
 | |
|         FLAGS.stub_network = False
 | |
| 
 | |
|         self.mox.ReplayAll()
 | |
| 
 | |
|         self.assertRaises(quantum_client.QuantumServerException,
 | |
|                           self.compute.run_instance,
 | |
|                           self.context,
 | |
|                           instance_id)
 | |
| 
 | |
|         instances = db.instance_get_all(context.get_admin_context())
 | |
|         self.assertEqual(vm_states.ERROR, instances[0]['vm_state'])
 | |
| 
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_network_is_deallocated_on_spawn_failure(self):
 | |
|         """When a spawn fails the network must be deallocated"""
 | |
|         instance_id = self._create_instance()
 | |
| 
 | |
|         self.mox.StubOutWithMock(self.compute, "_setup_block_device_mapping")
 | |
|         self.compute._setup_block_device_mapping(mox.IgnoreArg(),
 | |
|                                                  mox.IgnoreArg()).\
 | |
|             AndRaise(rpc.common.RemoteError('', '', ''))
 | |
| 
 | |
|         self.mox.ReplayAll()
 | |
| 
 | |
|         self.assertRaises(rpc.common.RemoteError,
 | |
|                           self.compute.run_instance,
 | |
|                           self.context,
 | |
|                           instance_id)
 | |
| 
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_lock(self):
 | |
|         """ensure locked instance cannot be changed"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
| 
 | |
|         non_admin_context = context.RequestContext(None, None, False, False)
 | |
| 
 | |
|         # decorator should return False (fail) with locked nonadmin context
 | |
|         self.compute.lock_instance(self.context, instance_id)
 | |
|         ret_val = self.compute.reboot_instance(non_admin_context, instance_id)
 | |
|         self.assertEqual(ret_val, False)
 | |
| 
 | |
|         # decorator should return None (success) with unlocked nonadmin context
 | |
|         self.compute.unlock_instance(self.context, instance_id)
 | |
|         ret_val = self.compute.reboot_instance(non_admin_context, instance_id)
 | |
|         self.assertEqual(ret_val, None)
 | |
| 
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_finish_resize(self):
 | |
|         """Contrived test to ensure finish_resize doesn't raise anything"""
 | |
| 
 | |
|         def fake(*args, **kwargs):
 | |
|             pass
 | |
| 
 | |
|         self.stubs.Set(self.compute.driver, 'finish_migration', fake)
 | |
|         self.stubs.Set(self.compute.network_api, 'get_instance_nw_info', fake)
 | |
|         context = self.context.elevated()
 | |
|         instance_id = self._create_instance()
 | |
|         instance_ref = db.instance_get(context, instance_id)
 | |
|         self.compute.prep_resize(context, instance_ref['uuid'], 1)
 | |
|         migration_ref = db.migration_get_by_instance_and_status(context,
 | |
|                 instance_ref['uuid'], 'pre-migrating')
 | |
|         try:
 | |
|             self.compute.finish_resize(context, instance_ref['uuid'],
 | |
|                     int(migration_ref['id']), {})
 | |
|         except KeyError, e:
 | |
|             # Only catch key errors. We want other reasons for the test to
 | |
|             # fail to actually error out so we don't obscure anything
 | |
|             self.fail()
 | |
| 
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_resize_instance_notification(self):
 | |
|         """Ensure notifications on instance migrate/resize"""
 | |
|         instance_id = self._create_instance()
 | |
|         context = self.context.elevated()
 | |
|         inst_ref = db.instance_get(context, instance_id)
 | |
| 
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         test_notifier.NOTIFICATIONS = []
 | |
| 
 | |
|         db.instance_update(self.context, instance_id, {'host': 'foo'})
 | |
|         self.compute.prep_resize(context, inst_ref['uuid'], 1)
 | |
|         migration_ref = db.migration_get_by_instance_and_status(context,
 | |
|                 inst_ref['uuid'], 'pre-migrating')
 | |
| 
 | |
|         self.assertEquals(len(test_notifier.NOTIFICATIONS), 1)
 | |
|         msg = test_notifier.NOTIFICATIONS[0]
 | |
|         self.assertEquals(msg['priority'], 'INFO')
 | |
|         self.assertEquals(msg['event_type'], 'compute.instance.resize.prep')
 | |
|         payload = msg['payload']
 | |
|         self.assertEquals(payload['tenant_id'], self.project_id)
 | |
|         self.assertEquals(payload['user_id'], self.user_id)
 | |
|         self.assertEquals(payload['instance_id'], inst_ref.uuid)
 | |
|         self.assertEquals(payload['instance_type'], 'm1.tiny')
 | |
|         type_id = instance_types.get_instance_type_by_name('m1.tiny')['id']
 | |
|         self.assertEquals(str(payload['instance_type_id']), str(type_id))
 | |
|         self.assertTrue('display_name' in payload)
 | |
|         self.assertTrue('created_at' in payload)
 | |
|         self.assertTrue('launched_at' in payload)
 | |
|         self.assertEquals(payload['image_ref'], '1')
 | |
|         self.compute.terminate_instance(context, instance_id)
 | |
| 
 | |
|     def test_resize_instance(self):
 | |
|         """Ensure instance can be migrated/resized"""
 | |
|         instance_id = self._create_instance()
 | |
|         context = self.context.elevated()
 | |
|         inst_ref = db.instance_get(context, instance_id)
 | |
| 
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         db.instance_update(self.context, inst_ref['uuid'],
 | |
|                            {'host': 'foo'})
 | |
|         self.compute.prep_resize(context, inst_ref['uuid'], 1)
 | |
|         migration_ref = db.migration_get_by_instance_and_status(context,
 | |
|                 inst_ref['uuid'], 'pre-migrating')
 | |
|         self.compute.resize_instance(context, inst_ref['uuid'],
 | |
|                 migration_ref['id'])
 | |
|         self.compute.terminate_instance(context, instance_id)
 | |
| 
 | |
|     def test_resize_invalid_flavor_fails(self):
 | |
|         """Ensure invalid flavors raise"""
 | |
|         instance_id = self._create_instance()
 | |
|         context = self.context.elevated()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
| 
 | |
|         self.assertRaises(exception.NotFound, self.compute_api.resize,
 | |
|                 context, instance_id, 200)
 | |
| 
 | |
|         self.compute.terminate_instance(context, instance_id)
 | |
| 
 | |
|     def test_resize_down_fails(self):
 | |
|         """Ensure resizing down raises and fails"""
 | |
|         context = self.context.elevated()
 | |
|         instance_id = self._create_instance()
 | |
| 
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         inst_type = instance_types.get_instance_type_by_name('m1.xlarge')
 | |
|         db.instance_update(self.context, instance_id,
 | |
|                 {'instance_type_id': inst_type['id']})
 | |
| 
 | |
|         self.assertRaises(exception.CannotResizeToSmallerSize,
 | |
|                           self.compute_api.resize, context, instance_id, 1)
 | |
| 
 | |
|         self.compute.terminate_instance(context, instance_id)
 | |
| 
 | |
|     def test_resize_same_size_fails(self):
 | |
|         """Ensure invalid flavors raise"""
 | |
|         context = self.context.elevated()
 | |
|         instance_id = self._create_instance()
 | |
| 
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
| 
 | |
|         self.assertRaises(exception.CannotResizeToSameSize,
 | |
|                           self.compute_api.resize, context, instance_id, 1)
 | |
| 
 | |
|         self.compute.terminate_instance(context, instance_id)
 | |
| 
 | |
|     def test_finish_revert_resize(self):
 | |
|         """Ensure that the flavor is reverted to the original on revert"""
 | |
|         context = self.context.elevated()
 | |
|         instance_id = self._create_instance()
 | |
| 
 | |
|         def fake(*args, **kwargs):
 | |
|             pass
 | |
| 
 | |
|         self.stubs.Set(self.compute.driver, 'finish_migration', fake)
 | |
|         self.stubs.Set(self.compute.driver, 'finish_revert_migration', fake)
 | |
|         self.stubs.Set(self.compute.network_api, 'get_instance_nw_info', fake)
 | |
| 
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
| 
 | |
|         # Confirm the instance size before the resize starts
 | |
|         inst_ref = db.instance_get(context, instance_id)
 | |
|         instance_type_ref = db.instance_type_get(context,
 | |
|                 inst_ref['instance_type_id'])
 | |
|         self.assertEqual(instance_type_ref['flavorid'], '1')
 | |
| 
 | |
|         db.instance_update(self.context, instance_id, {'host': 'foo'})
 | |
| 
 | |
|         new_instance_type_ref = db.instance_type_get_by_flavor_id(context, 3)
 | |
|         self.compute.prep_resize(context, inst_ref['uuid'],
 | |
|                                  new_instance_type_ref['id'])
 | |
| 
 | |
|         migration_ref = db.migration_get_by_instance_and_status(context,
 | |
|                 inst_ref['uuid'], 'pre-migrating')
 | |
| 
 | |
|         self.compute.resize_instance(context, inst_ref['uuid'],
 | |
|                 migration_ref['id'])
 | |
|         self.compute.finish_resize(context, inst_ref['uuid'],
 | |
|                     int(migration_ref['id']), {})
 | |
| 
 | |
|         # Prove that the instance size is now the new size
 | |
|         inst_ref = db.instance_get(context, instance_id)
 | |
|         instance_type_ref = db.instance_type_get(context,
 | |
|                 inst_ref['instance_type_id'])
 | |
|         self.assertEqual(instance_type_ref['flavorid'], '3')
 | |
| 
 | |
|         # Finally, revert and confirm the old flavor has been applied
 | |
|         self.compute.revert_resize(context, inst_ref['uuid'],
 | |
|                 migration_ref['id'])
 | |
|         self.compute.finish_revert_resize(context, inst_ref['uuid'],
 | |
|                 migration_ref['id'])
 | |
| 
 | |
|         inst_ref = db.instance_get(context, instance_id)
 | |
|         instance_type_ref = db.instance_type_get(context,
 | |
|                 inst_ref['instance_type_id'])
 | |
|         self.assertEqual(instance_type_ref['flavorid'], '1')
 | |
| 
 | |
|         self.compute.terminate_instance(context, instance_id)
 | |
| 
 | |
|     def test_get_by_flavor_id(self):
 | |
|         type = instance_types.get_instance_type_by_flavor_id(1)
 | |
|         self.assertEqual(type['name'], 'm1.tiny')
 | |
| 
 | |
|     def test_resize_same_source_fails(self):
 | |
|         """Ensure instance fails to migrate when source and destination are
 | |
|         the same host"""
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         inst_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertRaises(exception.Error, self.compute.prep_resize,
 | |
|                 self.context, inst_ref['uuid'], 1)
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_resize_instance_handles_migration_error(self):
 | |
|         """Ensure vm_state is ERROR when MigrationError occurs"""
 | |
|         def raise_migration_failure(*args):
 | |
|             raise exception.MigrationError(reason='test failure')
 | |
|         self.stubs.Set(self.compute.driver,
 | |
|                 'migrate_disk_and_power_off',
 | |
|                 raise_migration_failure)
 | |
| 
 | |
|         instance_id = self._create_instance()
 | |
|         context = self.context.elevated()
 | |
|         inst_ref = db.instance_get(context, instance_id)
 | |
| 
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         db.instance_update(self.context, inst_ref['uuid'], {'host': 'foo'})
 | |
|         self.compute.prep_resize(context, inst_ref['uuid'], 1)
 | |
|         migration_ref = db.migration_get_by_instance_and_status(context,
 | |
|                 inst_ref['uuid'], 'pre-migrating')
 | |
|         self.compute.resize_instance(context,
 | |
|                                      inst_ref['uuid'],
 | |
|                                      migration_ref['id'])
 | |
|         inst_ref = db.instance_get(context, instance_id)
 | |
|         self.assertEqual(inst_ref['vm_state'], vm_states.ERROR)
 | |
|         self.compute.terminate_instance(context, instance_id)
 | |
| 
 | |
|     def test_migrate(self):
 | |
|         context = self.context.elevated()
 | |
|         instance_id = self._create_instance()
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
|         # Migrate simply calls resize() without a flavor_id.
 | |
|         self.compute_api.resize(context, instance_id, None)
 | |
|         self.compute.terminate_instance(context, instance_id)
 | |
| 
 | |
|     def _setup_other_managers(self):
 | |
|         self.volume_manager = utils.import_object(FLAGS.volume_manager)
 | |
|         self.network_manager = utils.import_object(FLAGS.network_manager)
 | |
|         self.compute_driver = utils.import_object(FLAGS.compute_driver)
 | |
| 
 | |
|     def test_pre_live_migration_instance_has_no_fixed_ip(self):
 | |
|         """Confirm raising exception if instance doesn't have fixed_ip."""
 | |
|         # creating instance testdata
 | |
|         instance_id = self._create_instance({'host': 'dummy'})
 | |
|         c = context.get_admin_context()
 | |
|         inst_ref = db.instance_get(c, instance_id)
 | |
|         topic = db.queue_get_for(c, FLAGS.compute_topic, inst_ref['host'])
 | |
| 
 | |
|         # start test
 | |
|         self.assertRaises(exception.FixedIpNotFoundForInstance,
 | |
|                           self.compute.pre_live_migration,
 | |
|                           c, inst_ref['id'], time=FakeTime())
 | |
|         # cleanup
 | |
|         db.instance_destroy(c, instance_id)
 | |
| 
 | |
|     def test_pre_live_migration_works_correctly(self):
 | |
|         """Confirm setup_compute_volume is called when volume is mounted."""
 | |
|         # creating instance testdata
 | |
|         instance_id = self._create_instance({'host': 'dummy'})
 | |
|         c = context.get_admin_context()
 | |
|         inst_ref = db.instance_get(c, instance_id)
 | |
|         topic = db.queue_get_for(c, FLAGS.compute_topic, inst_ref['host'])
 | |
| 
 | |
|         # creating mocks
 | |
|         self.mox.StubOutWithMock(self.compute.driver, 'pre_live_migration')
 | |
|         self.compute.driver.pre_live_migration({'block_device_mapping': []})
 | |
|         dummy_nw_info = [[None, {'ips':'1.1.1.1'}]]
 | |
|         self.mox.StubOutWithMock(self.compute, '_get_instance_nw_info')
 | |
|         self.compute._get_instance_nw_info(c, mox.IsA(inst_ref)
 | |
|             ).AndReturn(dummy_nw_info)
 | |
|         self.mox.StubOutWithMock(self.compute.driver, 'plug_vifs')
 | |
|         self.compute.driver.plug_vifs(mox.IsA(inst_ref), dummy_nw_info)
 | |
|         self.mox.StubOutWithMock(self.compute.driver,
 | |
|                                  'ensure_filtering_rules_for_instance')
 | |
|         self.compute.driver.ensure_filtering_rules_for_instance(
 | |
|             mox.IsA(inst_ref), dummy_nw_info)
 | |
| 
 | |
|         # start test
 | |
|         self.mox.ReplayAll()
 | |
|         ret = self.compute.pre_live_migration(c, inst_ref['id'])
 | |
|         self.assertEqual(ret, None)
 | |
| 
 | |
|         # cleanup
 | |
|         db.instance_destroy(c, instance_id)
 | |
| 
 | |
|     def test_live_migration_dest_raises_exception(self):
 | |
|         """Confirm exception when pre_live_migration fails."""
 | |
|         # creating instance testdata
 | |
|         instance_id = self._create_instance({'host': 'dummy'})
 | |
|         c = context.get_admin_context()
 | |
|         inst_ref = db.instance_get(c, instance_id)
 | |
|         topic = db.queue_get_for(c, FLAGS.compute_topic, inst_ref['host'])
 | |
|         # creating volume testdata
 | |
|         volume_id = 1
 | |
|         db.volume_create(c, {'id': volume_id})
 | |
|         values = {'instance_id': instance_id, 'device_name': '/dev/vdc',
 | |
|             'delete_on_termination': False, 'volume_id': volume_id}
 | |
|         db.block_device_mapping_create(c, values)
 | |
| 
 | |
|         # creating mocks
 | |
|         self.mox.StubOutWithMock(rpc, 'call')
 | |
|         rpc.call(c, FLAGS.volume_topic, {"method": "check_for_export",
 | |
|                                          "args": {'instance_id': instance_id}})
 | |
|         rpc.call(c, topic, {"method": "pre_live_migration",
 | |
|                             "args": {'instance_id': instance_id,
 | |
|                                      'block_migration': True,
 | |
|                                      'disk': None}}).\
 | |
|                             AndRaise(rpc.common.RemoteError('', '', ''))
 | |
|         # mocks for rollback
 | |
|         rpc.call(c, topic, {"method": "remove_volume_connection",
 | |
|                             "args": {'instance_id': instance_id,
 | |
|                                      'volume_id': volume_id}})
 | |
|         rpc.cast(c, topic, {"method": "rollback_live_migration_at_destination",
 | |
|                             "args": {'instance_id': inst_ref['id']}})
 | |
| 
 | |
|         # start test
 | |
|         self.mox.ReplayAll()
 | |
|         self.assertRaises(rpc.RemoteError,
 | |
|                           self.compute.live_migration,
 | |
|                           c, instance_id, inst_ref['host'], True)
 | |
| 
 | |
|         # cleanup
 | |
|         for bdms in db.block_device_mapping_get_all_by_instance(c,
 | |
|                                                                 instance_id):
 | |
|             db.block_device_mapping_destroy(c, bdms['id'])
 | |
|         db.volume_destroy(c, volume_id)
 | |
|         db.instance_destroy(c, instance_id)
 | |
| 
 | |
|     def test_live_migration_works_correctly(self):
 | |
|         """Confirm live_migration() works as expected correctly."""
 | |
|         # creating instance testdata
 | |
|         instance_id = self._create_instance({'host': 'dummy'})
 | |
|         c = context.get_admin_context()
 | |
|         inst_ref = db.instance_get(c, instance_id)
 | |
|         topic = db.queue_get_for(c, FLAGS.compute_topic, inst_ref['host'])
 | |
| 
 | |
|         # create
 | |
|         self.mox.StubOutWithMock(rpc, 'call')
 | |
|         rpc.call(c, topic, {"method": "pre_live_migration",
 | |
|                             "args": {'instance_id': instance_id,
 | |
|                                      'block_migration': False,
 | |
|                                      'disk': None}})
 | |
| 
 | |
|         # start test
 | |
|         self.mox.ReplayAll()
 | |
|         ret = self.compute.live_migration(c, inst_ref['id'], inst_ref['host'])
 | |
|         self.assertEqual(ret, None)
 | |
| 
 | |
|         # cleanup
 | |
|         db.instance_destroy(c, instance_id)
 | |
| 
 | |
|     def test_post_live_migration_working_correctly(self):
 | |
|         """Confirm post_live_migration() works as expected correctly."""
 | |
|         dest = 'desthost'
 | |
|         flo_addr = '1.2.1.2'
 | |
| 
 | |
|         # creating testdata
 | |
|         c = context.get_admin_context()
 | |
|         instance_id = self._create_instance({'state_description': 'migrating',
 | |
|                                              'state': power_state.PAUSED})
 | |
|         i_ref = db.instance_get(c, instance_id)
 | |
|         db.instance_update(c, i_ref['id'], {'vm_state': vm_states.MIGRATING,
 | |
|                                             'power_state': power_state.PAUSED})
 | |
|         v_ref = db.volume_create(c, {'size': 1, 'instance_id': instance_id})
 | |
|         fix_addr = db.fixed_ip_create(c, {'address': '1.1.1.1',
 | |
|                                           'instance_id': instance_id})
 | |
|         fix_ref = db.fixed_ip_get_by_address(c, fix_addr)
 | |
|         flo_ref = db.floating_ip_create(c, {'address': flo_addr,
 | |
|                                         'fixed_ip_id': fix_ref['id']})
 | |
| 
 | |
|         # creating mocks
 | |
|         self.mox.StubOutWithMock(self.compute.driver, 'unfilter_instance')
 | |
|         self.compute.driver.unfilter_instance(i_ref, [])
 | |
|         self.mox.StubOutWithMock(rpc, 'call')
 | |
|         rpc.call(c, db.queue_get_for(c, FLAGS.compute_topic, dest),
 | |
|             {"method": "post_live_migration_at_destination",
 | |
|              "args": {'instance_id': i_ref['id'], 'block_migration': False}})
 | |
| 
 | |
|         # start test
 | |
|         self.mox.ReplayAll()
 | |
|         ret = self.compute.post_live_migration(c, i_ref, dest)
 | |
| 
 | |
|         # make sure every data is rewritten to destinatioin hostname.
 | |
|         i_ref = db.instance_get(c, i_ref['id'])
 | |
|         c1 = (i_ref['host'] == dest)
 | |
|         flo_refs = db.floating_ip_get_all_by_host(c, dest)
 | |
|         c2 = (len(flo_refs) != 0 and flo_refs[0]['address'] == flo_addr)
 | |
|         self.assertTrue(c1 and c2)
 | |
| 
 | |
|         # cleanup
 | |
|         db.instance_destroy(c, instance_id)
 | |
|         db.volume_destroy(c, v_ref['id'])
 | |
|         db.floating_ip_destroy(c, flo_addr)
 | |
| 
 | |
|     def test_run_kill_vm(self):
 | |
|         """Detect when a vm is terminated behind the scenes"""
 | |
|         self.stubs.Set(compute_manager.ComputeManager,
 | |
|                 '_report_driver_status', nop_report_driver_status)
 | |
| 
 | |
|         instance_id = self._create_instance()
 | |
| 
 | |
|         self.compute.run_instance(self.context, instance_id)
 | |
| 
 | |
|         instances = db.instance_get_all(context.get_admin_context())
 | |
|         LOG.info(_("Running instances: %s"), instances)
 | |
|         self.assertEqual(len(instances), 1)
 | |
| 
 | |
|         instance_name = instances[0].name
 | |
|         self.compute.driver.test_remove_vm(instance_name)
 | |
| 
 | |
|         # Force the compute manager to do its periodic poll
 | |
|         error_list = self.compute.periodic_tasks(context.get_admin_context())
 | |
|         self.assertFalse(error_list)
 | |
| 
 | |
|         instances = db.instance_get_all(context.get_admin_context())
 | |
|         LOG.info(_("After force-killing instances: %s"), instances)
 | |
|         self.assertEqual(len(instances), 1)
 | |
|         self.assertEqual(power_state.NOSTATE, instances[0]['power_state'])
 | |
| 
 | |
|     def test_get_all_by_name_regexp(self):
 | |
|         """Test searching instances by name (display_name)"""
 | |
|         c = context.get_admin_context()
 | |
|         instance_id1 = self._create_instance({'display_name': 'woot'})
 | |
|         instance_id2 = self._create_instance({
 | |
|                 'display_name': 'woo',
 | |
|                 'id': 20})
 | |
|         instance_id3 = self._create_instance({
 | |
|                 'display_name': 'not-woot',
 | |
|                 'id': 30})
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'name': 'woo.*'})
 | |
|         self.assertEqual(len(instances), 2)
 | |
|         instance_ids = [instance['id'] for instance in instances]
 | |
|         self.assertTrue(instance_id1 in instance_ids)
 | |
|         self.assertTrue(instance_id2 in instance_ids)
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'name': 'woot.*'})
 | |
|         instance_ids = [instance['id'] for instance in instances]
 | |
|         self.assertEqual(len(instances), 1)
 | |
|         self.assertTrue(instance_id1 in instance_ids)
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'name': '.*oot.*'})
 | |
|         self.assertEqual(len(instances), 2)
 | |
|         instance_ids = [instance['id'] for instance in instances]
 | |
|         self.assertTrue(instance_id1 in instance_ids)
 | |
|         self.assertTrue(instance_id3 in instance_ids)
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'name': 'n.*'})
 | |
|         self.assertEqual(len(instances), 1)
 | |
|         instance_ids = [instance['id'] for instance in instances]
 | |
|         self.assertTrue(instance_id3 in instance_ids)
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'name': 'noth.*'})
 | |
|         self.assertEqual(len(instances), 0)
 | |
| 
 | |
|         db.instance_destroy(c, instance_id1)
 | |
|         db.instance_destroy(c, instance_id2)
 | |
|         db.instance_destroy(c, instance_id3)
 | |
| 
 | |
|     def test_get_all_by_instance_name_regexp(self):
 | |
|         """Test searching instances by name"""
 | |
|         self.flags(instance_name_template='instance-%d')
 | |
| 
 | |
|         c = context.get_admin_context()
 | |
|         instance_id1 = self._create_instance()
 | |
|         instance_id2 = self._create_instance({'id': 2})
 | |
|         instance_id3 = self._create_instance({'id': 10})
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'instance_name': 'instance.*'})
 | |
|         self.assertEqual(len(instances), 3)
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'instance_name': '.*\-\d$'})
 | |
|         self.assertEqual(len(instances), 2)
 | |
|         instance_ids = [instance['id'] for instance in instances]
 | |
|         self.assertTrue(instance_id1 in instance_ids)
 | |
|         self.assertTrue(instance_id2 in instance_ids)
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'instance_name': 'i.*2'})
 | |
|         self.assertEqual(len(instances), 1)
 | |
|         self.assertEqual(instances[0]['id'], instance_id2)
 | |
| 
 | |
|         db.instance_destroy(c, instance_id1)
 | |
|         db.instance_destroy(c, instance_id2)
 | |
|         db.instance_destroy(c, instance_id3)
 | |
| 
 | |
|     def test_get_all_by_multiple_options_at_once(self):
 | |
|         """Test searching by multiple options at once"""
 | |
|         c = context.get_admin_context()
 | |
|         network_manager = fake_network.FakeNetworkManager()
 | |
|         self.stubs.Set(self.compute_api.network_api,
 | |
|                        'get_instance_uuids_by_ip_filter',
 | |
|                        network_manager.get_instance_uuids_by_ip_filter)
 | |
|         self.stubs.Set(network_manager.db,
 | |
|                        'instance_get_id_to_uuid_mapping',
 | |
|                        db.instance_get_id_to_uuid_mapping)
 | |
| 
 | |
|         instance_id1 = self._create_instance({'display_name': 'woot',
 | |
|                                               'id': 0})
 | |
|         instance_id2 = self._create_instance({
 | |
|                 'display_name': 'woo',
 | |
|                 'id': 20})
 | |
|         instance_id3 = self._create_instance({
 | |
|                 'display_name': 'not-woot',
 | |
|                 'id': 30})
 | |
| 
 | |
|         # ip ends up matching 2nd octet here.. so all 3 match ip
 | |
|         # but 'name' only matches one
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'ip': '.*\.1', 'name': 'not.*'})
 | |
|         self.assertEqual(len(instances), 1)
 | |
|         self.assertEqual(instances[0]['id'], instance_id3)
 | |
| 
 | |
|         # ip ends up matching any ip with a '1' in the last octet..
 | |
|         # so instance 1 and 3.. but name should only match #1
 | |
|         # but 'name' only matches one
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'ip': '.*\.1$', 'name': '^woo.*'})
 | |
|         self.assertEqual(len(instances), 1)
 | |
|         self.assertEqual(instances[0]['id'], instance_id1)
 | |
| 
 | |
|         # same as above but no match on name (name matches instance_id1
 | |
|         # but the ip query doesn't
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'ip': '.*\.2$', 'name': '^woot.*'})
 | |
|         self.assertEqual(len(instances), 0)
 | |
| 
 | |
|         # ip matches all 3... ipv6 matches #2+#3...name matches #3
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'ip': '.*\.1',
 | |
|                              'name': 'not.*',
 | |
|                              'ip6': '^.*12.*34.*'})
 | |
|         self.assertEqual(len(instances), 1)
 | |
|         self.assertEqual(instances[0]['id'], instance_id3)
 | |
| 
 | |
|         db.instance_destroy(c, instance_id1)
 | |
|         db.instance_destroy(c, instance_id2)
 | |
|         db.instance_destroy(c, instance_id3)
 | |
| 
 | |
|     def test_get_all_by_image(self):
 | |
|         """Test searching instances by image"""
 | |
| 
 | |
|         c = context.get_admin_context()
 | |
|         instance_id1 = self._create_instance({'image_ref': '1234'})
 | |
|         instance_id2 = self._create_instance({
 | |
|             'id': 2,
 | |
|             'image_ref': '4567'})
 | |
|         instance_id3 = self._create_instance({
 | |
|             'id': 10,
 | |
|             'image_ref': '4567'})
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'image': '123'})
 | |
|         self.assertEqual(len(instances), 0)
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'image': '1234'})
 | |
|         self.assertEqual(len(instances), 1)
 | |
|         self.assertEqual(instances[0]['id'], instance_id1)
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'image': '4567'})
 | |
|         self.assertEqual(len(instances), 2)
 | |
|         instance_ids = [instance['id'] for instance in instances]
 | |
|         self.assertTrue(instance_id2 in instance_ids)
 | |
|         self.assertTrue(instance_id3 in instance_ids)
 | |
| 
 | |
|         # Test passing a list as search arg
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'image': ['1234', '4567']})
 | |
|         self.assertEqual(len(instances), 3)
 | |
| 
 | |
|         db.instance_destroy(c, instance_id1)
 | |
|         db.instance_destroy(c, instance_id2)
 | |
|         db.instance_destroy(c, instance_id3)
 | |
| 
 | |
|     def test_get_all_by_flavor(self):
 | |
|         """Test searching instances by image"""
 | |
| 
 | |
|         c = context.get_admin_context()
 | |
|         instance_id1 = self._create_instance({'instance_type_id': 1})
 | |
|         instance_id2 = self._create_instance({
 | |
|                 'id': 2,
 | |
|                 'instance_type_id': 2})
 | |
|         instance_id3 = self._create_instance({
 | |
|                 'id': 10,
 | |
|                 'instance_type_id': 2})
 | |
| 
 | |
|         # NOTE(comstud): Migrations set up the instance_types table
 | |
|         # for us.  Therefore, we assume the following is true for
 | |
|         # these tests:
 | |
|         # instance_type_id 1 == flavor 3
 | |
|         # instance_type_id 2 == flavor 1
 | |
|         # instance_type_id 3 == flavor 4
 | |
|         # instance_type_id 4 == flavor 5
 | |
|         # instance_type_id 5 == flavor 2
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'flavor': 5})
 | |
|         self.assertEqual(len(instances), 0)
 | |
| 
 | |
|         self.assertRaises(exception.FlavorNotFound,
 | |
|                 self.compute_api.get_all,
 | |
|                 c, search_opts={'flavor': 99})
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'flavor': 3})
 | |
|         self.assertEqual(len(instances), 1)
 | |
|         self.assertEqual(instances[0]['id'], instance_id1)
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'flavor': 1})
 | |
|         self.assertEqual(len(instances), 2)
 | |
|         instance_ids = [instance['id'] for instance in instances]
 | |
|         self.assertTrue(instance_id2 in instance_ids)
 | |
|         self.assertTrue(instance_id3 in instance_ids)
 | |
| 
 | |
|         db.instance_destroy(c, instance_id1)
 | |
|         db.instance_destroy(c, instance_id2)
 | |
|         db.instance_destroy(c, instance_id3)
 | |
| 
 | |
|     def test_get_all_by_state(self):
 | |
|         """Test searching instances by state"""
 | |
| 
 | |
|         c = context.get_admin_context()
 | |
|         instance_id1 = self._create_instance({
 | |
|             'power_state': power_state.SHUTDOWN,
 | |
|         })
 | |
|         instance_id2 = self._create_instance({
 | |
|             'id': 2,
 | |
|             'power_state': power_state.RUNNING,
 | |
|         })
 | |
|         instance_id3 = self._create_instance({
 | |
|             'id': 10,
 | |
|             'power_state': power_state.RUNNING,
 | |
|         })
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'power_state': power_state.SUSPENDED})
 | |
|         self.assertEqual(len(instances), 0)
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'power_state': power_state.SHUTDOWN})
 | |
|         self.assertEqual(len(instances), 1)
 | |
|         self.assertEqual(instances[0]['id'], instance_id1)
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'power_state': power_state.RUNNING})
 | |
|         self.assertEqual(len(instances), 2)
 | |
|         instance_ids = [instance['id'] for instance in instances]
 | |
|         self.assertTrue(instance_id2 in instance_ids)
 | |
|         self.assertTrue(instance_id3 in instance_ids)
 | |
| 
 | |
|         # Test passing a list as search arg
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'power_state': [power_state.SHUTDOWN,
 | |
|                         power_state.RUNNING]})
 | |
|         self.assertEqual(len(instances), 3)
 | |
| 
 | |
|         db.instance_destroy(c, instance_id1)
 | |
|         db.instance_destroy(c, instance_id2)
 | |
|         db.instance_destroy(c, instance_id3)
 | |
| 
 | |
|     def test_get_all_by_metadata(self):
 | |
|         """Test searching instances by metadata"""
 | |
| 
 | |
|         c = context.get_admin_context()
 | |
|         instance_id0 = self._create_instance()
 | |
|         instance_id1 = self._create_instance({
 | |
|                 'metadata': {'key1': 'value1'}})
 | |
|         instance_id2 = self._create_instance({
 | |
|                 'metadata': {'key2': 'value2'}})
 | |
|         instance_id3 = self._create_instance({
 | |
|                 'metadata': {'key3': 'value3'}})
 | |
|         instance_id4 = self._create_instance({
 | |
|                 'metadata': {'key3': 'value3',
 | |
|                              'key4': 'value4'}})
 | |
| 
 | |
|         # get all instances
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'metadata': {}})
 | |
|         self.assertEqual(len(instances), 5)
 | |
| 
 | |
|         # wrong key/value combination
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'metadata': {'key1': 'value3'}})
 | |
|         self.assertEqual(len(instances), 0)
 | |
| 
 | |
|         # non-existing keys
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'metadata': {'key5': 'value1'}})
 | |
|         self.assertEqual(len(instances), 0)
 | |
| 
 | |
|         # find existing instance
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'metadata': {'key2': 'value2'}})
 | |
|         self.assertEqual(len(instances), 1)
 | |
|         self.assertEqual(instances[0]['id'], instance_id2)
 | |
| 
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'metadata': {'key3': 'value3'}})
 | |
|         self.assertEqual(len(instances), 2)
 | |
|         instance_ids = [instance['id'] for instance in instances]
 | |
|         self.assertTrue(instance_id3 in instance_ids)
 | |
|         self.assertTrue(instance_id4 in instance_ids)
 | |
| 
 | |
|         # multiple criterias as a dict
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'metadata': {'key3': 'value3',
 | |
|                                           'key4': 'value4'}})
 | |
|         self.assertEqual(len(instances), 1)
 | |
|         self.assertEqual(instances[0]['id'], instance_id4)
 | |
| 
 | |
|         # multiple criterias as a list
 | |
|         instances = self.compute_api.get_all(c,
 | |
|                 search_opts={'metadata': [{'key4': 'value4'},
 | |
|                                           {'key3': 'value3'}]})
 | |
|         self.assertEqual(len(instances), 1)
 | |
|         self.assertEqual(instances[0]['id'], instance_id4)
 | |
| 
 | |
|         db.instance_destroy(c, instance_id0)
 | |
|         db.instance_destroy(c, instance_id1)
 | |
|         db.instance_destroy(c, instance_id2)
 | |
|         db.instance_destroy(c, instance_id3)
 | |
|         db.instance_destroy(c, instance_id4)
 | |
| 
 | |
|     @staticmethod
 | |
|     def _parse_db_block_device_mapping(bdm_ref):
 | |
|         attr_list = ('delete_on_termination', 'device_name', 'no_device',
 | |
|                      'virtual_name', 'volume_id', 'volume_size', 'snapshot_id')
 | |
|         bdm = {}
 | |
|         for attr in attr_list:
 | |
|             val = bdm_ref.get(attr, None)
 | |
|             if val:
 | |
|                 bdm[attr] = val
 | |
| 
 | |
|         return bdm
 | |
| 
 | |
|     def test_update_block_device_mapping(self):
 | |
|         swap_size = 1
 | |
|         instance_type = {'swap': swap_size}
 | |
|         instance_id = self._create_instance()
 | |
|         mappings = [
 | |
|                 {'virtual': 'ami', 'device': 'sda1'},
 | |
|                 {'virtual': 'root', 'device': '/dev/sda1'},
 | |
| 
 | |
|                 {'virtual': 'swap', 'device': 'sdb4'},
 | |
|                 {'virtual': 'swap', 'device': 'sdb3'},
 | |
|                 {'virtual': 'swap', 'device': 'sdb2'},
 | |
|                 {'virtual': 'swap', 'device': 'sdb1'},
 | |
| 
 | |
|                 {'virtual': 'ephemeral0', 'device': 'sdc1'},
 | |
|                 {'virtual': 'ephemeral1', 'device': 'sdc2'},
 | |
|                 {'virtual': 'ephemeral2', 'device': 'sdc3'}]
 | |
|         block_device_mapping = [
 | |
|                 # root
 | |
|                 {'device_name': '/dev/sda1',
 | |
|                  'snapshot_id': 0x12345678,
 | |
|                  'delete_on_termination': False},
 | |
| 
 | |
| 
 | |
|                 # overwrite swap
 | |
|                 {'device_name': '/dev/sdb2',
 | |
|                  'snapshot_id': 0x23456789,
 | |
|                  'delete_on_termination': False},
 | |
|                 {'device_name': '/dev/sdb3',
 | |
|                  'snapshot_id': 0x3456789A},
 | |
|                 {'device_name': '/dev/sdb4',
 | |
|                  'no_device': True},
 | |
| 
 | |
|                 # overwrite ephemeral
 | |
|                 {'device_name': '/dev/sdc2',
 | |
|                  'snapshot_id': 0x456789AB,
 | |
|                  'delete_on_termination': False},
 | |
|                 {'device_name': '/dev/sdc3',
 | |
|                  'snapshot_id': 0x56789ABC},
 | |
|                 {'device_name': '/dev/sdc4',
 | |
|                  'no_device': True},
 | |
| 
 | |
|                 # volume
 | |
|                 {'device_name': '/dev/sdd1',
 | |
|                  'snapshot_id': 0x87654321,
 | |
|                  'delete_on_termination': False},
 | |
|                 {'device_name': '/dev/sdd2',
 | |
|                  'snapshot_id': 0x98765432},
 | |
|                 {'device_name': '/dev/sdd3',
 | |
|                  'snapshot_id': 0xA9875463},
 | |
|                 {'device_name': '/dev/sdd4',
 | |
|                  'no_device': True}]
 | |
| 
 | |
|         self.compute_api._update_image_block_device_mapping(
 | |
|             self.context, instance_type, instance_id, mappings)
 | |
| 
 | |
|         bdms = [self._parse_db_block_device_mapping(bdm_ref)
 | |
|                 for bdm_ref in db.block_device_mapping_get_all_by_instance(
 | |
|                     self.context, instance_id)]
 | |
|         expected_result = [
 | |
|             {'virtual_name': 'swap', 'device_name': '/dev/sdb1',
 | |
|              'volume_size': swap_size},
 | |
|             {'virtual_name': 'ephemeral0', 'device_name': '/dev/sdc1'},
 | |
| 
 | |
|             # NOTE(yamahata): ATM only ephemeral0 is supported.
 | |
|             #                 they're ignored for now
 | |
|             #{'virtual_name': 'ephemeral1', 'device_name': '/dev/sdc2'},
 | |
|             #{'virtual_name': 'ephemeral2', 'device_name': '/dev/sdc3'}
 | |
|             ]
 | |
|         bdms.sort()
 | |
|         expected_result.sort()
 | |
|         self.assertDictListMatch(bdms, expected_result)
 | |
| 
 | |
|         self.compute_api._update_block_device_mapping(
 | |
|             self.context, instance_types.get_default_instance_type(),
 | |
|             instance_id, block_device_mapping)
 | |
|         bdms = [self._parse_db_block_device_mapping(bdm_ref)
 | |
|                 for bdm_ref in db.block_device_mapping_get_all_by_instance(
 | |
|                     self.context, instance_id)]
 | |
|         expected_result = [
 | |
|             {'snapshot_id': 0x12345678, 'device_name': '/dev/sda1'},
 | |
| 
 | |
|             {'virtual_name': 'swap', 'device_name': '/dev/sdb1',
 | |
|              'volume_size': swap_size},
 | |
|             {'snapshot_id': 0x23456789, 'device_name': '/dev/sdb2'},
 | |
|             {'snapshot_id': 0x3456789A, 'device_name': '/dev/sdb3'},
 | |
|             {'no_device': True, 'device_name': '/dev/sdb4'},
 | |
| 
 | |
|             {'virtual_name': 'ephemeral0', 'device_name': '/dev/sdc1'},
 | |
|             {'snapshot_id': 0x456789AB, 'device_name': '/dev/sdc2'},
 | |
|             {'snapshot_id': 0x56789ABC, 'device_name': '/dev/sdc3'},
 | |
|             {'no_device': True, 'device_name': '/dev/sdc4'},
 | |
| 
 | |
|             {'snapshot_id': 0x87654321, 'device_name': '/dev/sdd1'},
 | |
|             {'snapshot_id': 0x98765432, 'device_name': '/dev/sdd2'},
 | |
|             {'snapshot_id': 0xA9875463, 'device_name': '/dev/sdd3'},
 | |
|             {'no_device': True, 'device_name': '/dev/sdd4'}]
 | |
|         bdms.sort()
 | |
|         expected_result.sort()
 | |
|         self.assertDictListMatch(bdms, expected_result)
 | |
| 
 | |
|         for bdm in db.block_device_mapping_get_all_by_instance(
 | |
|             self.context, instance_id):
 | |
|             db.block_device_mapping_destroy(self.context, bdm['id'])
 | |
|         self.compute.terminate_instance(self.context, instance_id)
 | |
| 
 | |
|     def test_volume_size(self):
 | |
|         local_size = 2
 | |
|         swap_size = 3
 | |
|         inst_type = {'local_gb': local_size, 'swap': swap_size}
 | |
|         self.assertEqual(self.compute_api._volume_size(inst_type,
 | |
|                                                           'ephemeral0'),
 | |
|                          local_size)
 | |
|         self.assertEqual(self.compute_api._volume_size(inst_type,
 | |
|                                                        'ephemeral1'),
 | |
|                          0)
 | |
|         self.assertEqual(self.compute_api._volume_size(inst_type,
 | |
|                                                        'swap'),
 | |
|                          swap_size)
 | |
| 
 | |
|     def test_reservation_id_one_instance(self):
 | |
|         """Verify building an instance has a reservation_id that
 | |
|         matches return value from create"""
 | |
|         (refs, resv_id) = self.compute_api.create(self.context,
 | |
|                 instance_types.get_default_instance_type(), None)
 | |
|         try:
 | |
|             self.assertEqual(len(refs), 1)
 | |
|             self.assertEqual(refs[0]['reservation_id'], resv_id)
 | |
|         finally:
 | |
|             db.instance_destroy(self.context, refs[0]['id'])
 | |
| 
 | |
|     def test_reservation_ids_two_instances(self):
 | |
|         """Verify building 2 instances at once results in a
 | |
|         reservation_id being returned equal to reservation id set
 | |
|         in both instances
 | |
|         """
 | |
|         (refs, resv_id) = self.compute_api.create(self.context,
 | |
|                 instance_types.get_default_instance_type(), None,
 | |
|                 min_count=2, max_count=2)
 | |
|         try:
 | |
|             self.assertEqual(len(refs), 2)
 | |
|             self.assertNotEqual(resv_id, None)
 | |
|         finally:
 | |
|             for instance in refs:
 | |
|                 self.assertEqual(instance['reservation_id'], resv_id)
 | |
|                 db.instance_destroy(self.context, instance['id'])
 | |
| 
 | |
|     def test_create_with_specified_reservation_id(self):
 | |
|         """Verify building instances with a specified
 | |
|         reservation_id results in the correct reservation_id
 | |
|         being set
 | |
|         """
 | |
| 
 | |
|         # We need admin context to be able to specify our own
 | |
|         # reservation_ids.
 | |
|         context = self.context.elevated()
 | |
|         # 1 instance
 | |
|         (refs, resv_id) = self.compute_api.create(context,
 | |
|                 instance_types.get_default_instance_type(), None,
 | |
|                 min_count=1, max_count=1, reservation_id='meow')
 | |
|         try:
 | |
|             self.assertEqual(len(refs), 1)
 | |
|             self.assertEqual(resv_id, 'meow')
 | |
|         finally:
 | |
|             self.assertEqual(refs[0]['reservation_id'], resv_id)
 | |
|             db.instance_destroy(self.context, refs[0]['id'])
 | |
| 
 | |
|         # 2 instances
 | |
|         (refs, resv_id) = self.compute_api.create(context,
 | |
|                 instance_types.get_default_instance_type(), None,
 | |
|                 min_count=2, max_count=2, reservation_id='woof')
 | |
|         try:
 | |
|             self.assertEqual(len(refs), 2)
 | |
|             self.assertEqual(resv_id, 'woof')
 | |
|         finally:
 | |
|             for instance in refs:
 | |
|                 self.assertEqual(instance['reservation_id'], resv_id)
 | |
|                 db.instance_destroy(self.context, instance['id'])
 | |
| 
 | |
|     def test_instance_name_template(self):
 | |
|         """Test the instance_name template"""
 | |
|         self.flags(instance_name_template='instance-%d')
 | |
|         instance_id = self._create_instance()
 | |
|         i_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertEqual(i_ref['name'], 'instance-%d' % i_ref['id'])
 | |
|         db.instance_destroy(self.context, i_ref['id'])
 | |
| 
 | |
|         self.flags(instance_name_template='instance-%(uuid)s')
 | |
|         instance_id = self._create_instance()
 | |
|         i_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertEqual(i_ref['name'], 'instance-%s' % i_ref['uuid'])
 | |
|         db.instance_destroy(self.context, i_ref['id'])
 | |
| 
 | |
|         self.flags(instance_name_template='%(id)d-%(uuid)s')
 | |
|         instance_id = self._create_instance()
 | |
|         i_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertEqual(i_ref['name'], '%d-%s' %
 | |
|                 (i_ref['id'], i_ref['uuid']))
 | |
|         db.instance_destroy(self.context, i_ref['id'])
 | |
| 
 | |
|         # not allowed.. default is uuid
 | |
|         self.flags(instance_name_template='%(name)s')
 | |
|         instance_id = self._create_instance()
 | |
|         i_ref = db.instance_get(self.context, instance_id)
 | |
|         self.assertEqual(i_ref['name'], i_ref['uuid'])
 | |
|         db.instance_destroy(self.context, i_ref['id'])
 | |
| 
 | |
| 
 | |
| class ComputeTestMinRamMinDisk(test.TestCase):
 | |
|     def setUp(self):
 | |
|         super(ComputeTestMinRamMinDisk, self).setUp()
 | |
|         self.compute = utils.import_object(FLAGS.compute_manager)
 | |
|         self.compute_api = compute.API()
 | |
|         self.context = context.RequestContext('fake', 'fake')
 | |
|         self.stubs.Set(rpc, 'call', rpc_call_wrapper)
 | |
|         self.stubs.Set(rpc, 'cast', rpc_cast_wrapper)
 | |
|         self.fake_image = {
 | |
|             'id': 1, 'properties': {'kernel_id': 1, 'ramdisk_id': 1}}
 | |
| 
 | |
|     def test_create_with_too_little_ram(self):
 | |
|         """Test an instance type with too little memory"""
 | |
| 
 | |
|         inst_type = instance_types.get_default_instance_type()
 | |
|         inst_type['memory_mb'] = 1
 | |
| 
 | |
|         def fake_show(*args):
 | |
|             img = copy(self.fake_image)
 | |
|             img['min_ram'] = 2
 | |
|             return img
 | |
|         self.stubs.Set(fake_image._FakeImageService, 'show', fake_show)
 | |
| 
 | |
|         self.assertRaises(exception.InstanceTypeMemoryTooSmall,
 | |
|             self.compute_api.create, self.context, inst_type, None)
 | |
| 
 | |
|         # Now increase the inst_type memory and make sure all is fine.
 | |
|         inst_type['memory_mb'] = 2
 | |
|         (refs, resv_id) = self.compute_api.create(self.context,
 | |
|                 inst_type, None)
 | |
|         db.instance_destroy(self.context, refs[0]['id'])
 | |
| 
 | |
|     def test_create_with_too_little_disk(self):
 | |
|         """Test an instance type with too little disk space"""
 | |
| 
 | |
|         inst_type = instance_types.get_default_instance_type()
 | |
|         inst_type['local_gb'] = 1
 | |
| 
 | |
|         def fake_show(*args):
 | |
|             img = copy(self.fake_image)
 | |
|             img['min_disk'] = 2
 | |
|             return img
 | |
|         self.stubs.Set(fake_image._FakeImageService, 'show', fake_show)
 | |
| 
 | |
|         self.assertRaises(exception.InstanceTypeDiskTooSmall,
 | |
|             self.compute_api.create, self.context, inst_type, None)
 | |
| 
 | |
|         # Now increase the inst_type disk space and make sure all is fine.
 | |
|         inst_type['local_gb'] = 2
 | |
|         (refs, resv_id) = self.compute_api.create(self.context,
 | |
|                 inst_type, None)
 | |
|         db.instance_destroy(self.context, refs[0]['id'])
 | |
| 
 | |
|     def test_create_just_enough_ram_and_disk(self):
 | |
|         """Test an instance type with just enough ram and disk space"""
 | |
| 
 | |
|         inst_type = instance_types.get_default_instance_type()
 | |
|         inst_type['local_gb'] = 2
 | |
|         inst_type['memory_mb'] = 2
 | |
| 
 | |
|         def fake_show(*args):
 | |
|             img = copy(self.fake_image)
 | |
|             img['min_ram'] = 2
 | |
|             img['min_disk'] = 2
 | |
|             return img
 | |
|         self.stubs.Set(fake_image._FakeImageService, 'show', fake_show)
 | |
| 
 | |
|         (refs, resv_id) = self.compute_api.create(self.context,
 | |
|                 inst_type, None)
 | |
|         db.instance_destroy(self.context, refs[0]['id'])
 | |
| 
 | |
|     def test_create_with_no_ram_and_disk_reqs(self):
 | |
|         """Test an instance type with no min_ram or min_disk"""
 | |
| 
 | |
|         inst_type = instance_types.get_default_instance_type()
 | |
|         inst_type['local_gb'] = 1
 | |
|         inst_type['memory_mb'] = 1
 | |
| 
 | |
|         def fake_show(*args):
 | |
|             return copy(self.fake_image)
 | |
|         self.stubs.Set(fake_image._FakeImageService, 'show', fake_show)
 | |
| 
 | |
|         (refs, resv_id) = self.compute_api.create(self.context,
 | |
|                 inst_type, None)
 | |
|         db.instance_destroy(self.context, refs[0]['id'])
 | 
