Merge trunk
Conflicts solved
This commit is contained in:
		
							
								
								
									
										49
									
								
								bin/nova-api
									
									
									
									
									
								
							
							
						
						
									
										49
									
								
								bin/nova-api
									
									
									
									
									
								
							| @@ -36,51 +36,15 @@ gettext.install('nova', unicode=1) | ||||
|  | ||||
| from nova import flags | ||||
| from nova import log as logging | ||||
| from nova import service | ||||
| from nova import utils | ||||
| from nova import version | ||||
| from nova import wsgi | ||||
|  | ||||
|  | ||||
| LOG = logging.getLogger('nova.api') | ||||
|  | ||||
| FLAGS = flags.FLAGS | ||||
| flags.DEFINE_string('paste_config', "api-paste.ini", | ||||
|                     'File name for the paste.deploy config for nova-api') | ||||
| flags.DEFINE_string('ec2_listen', "0.0.0.0", | ||||
|                     'IP address for EC2 API to listen') | ||||
| flags.DEFINE_integer('ec2_listen_port', 8773, 'port for ec2 api to listen') | ||||
| flags.DEFINE_string('osapi_listen', "0.0.0.0", | ||||
|                     'IP address for OpenStack API to listen') | ||||
| flags.DEFINE_integer('osapi_listen_port', 8774, 'port for os api to listen') | ||||
| flags.DEFINE_flag(flags.HelpFlag()) | ||||
| flags.DEFINE_flag(flags.HelpshortFlag()) | ||||
| flags.DEFINE_flag(flags.HelpXMLFlag()) | ||||
|  | ||||
| API_ENDPOINTS = ['ec2', 'osapi'] | ||||
|  | ||||
|  | ||||
| def run_app(paste_config_file): | ||||
|     LOG.debug(_("Using paste.deploy config at: %s"), paste_config_file) | ||||
|     apps = [] | ||||
|     for api in API_ENDPOINTS: | ||||
|         config = wsgi.load_paste_configuration(paste_config_file, api) | ||||
|         if config is None: | ||||
|             LOG.debug(_("No paste configuration for app: %s"), api) | ||||
|             continue | ||||
|         LOG.debug(_("App Config: %(api)s\n%(config)r") % locals()) | ||||
|         LOG.info(_("Running %s API"), api) | ||||
|         app = wsgi.load_paste_app(paste_config_file, api) | ||||
|         apps.append((app, getattr(FLAGS, "%s_listen_port" % api), | ||||
|                      getattr(FLAGS, "%s_listen" % api))) | ||||
|     if len(apps) == 0: | ||||
|         LOG.error(_("No known API applications configured in %s."), | ||||
|                   paste_config_file) | ||||
|         return | ||||
|  | ||||
|     server = wsgi.Server() | ||||
|     for app in apps: | ||||
|         server.start(*app) | ||||
|     server.wait() | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     utils.default_flagfile() | ||||
| @@ -92,9 +56,6 @@ if __name__ == '__main__': | ||||
|     for flag in FLAGS: | ||||
|         flag_get = FLAGS.get(flag, None) | ||||
|         LOG.debug("%(flag)s : %(flag_get)s" % locals()) | ||||
|     conf = wsgi.paste_config_file(FLAGS.paste_config) | ||||
|     if conf: | ||||
|         run_app(conf) | ||||
|     else: | ||||
|         LOG.error(_("No paste configuration found for: %s"), | ||||
|                   FLAGS.paste_config) | ||||
|  | ||||
|     service = service.serve_wsgi(service.ApiService) | ||||
|     service.wait() | ||||
|   | ||||
							
								
								
									
										166
									
								
								bin/nova-manage
									
									
									
									
									
								
							
							
						
						
									
										166
									
								
								bin/nova-manage
									
									
									
									
									
								
							| @@ -55,6 +55,8 @@ | ||||
|  | ||||
| import datetime | ||||
| import gettext | ||||
| import glob | ||||
| import json | ||||
| import os | ||||
| import re | ||||
| import sys | ||||
| @@ -81,7 +83,7 @@ from nova import log as logging | ||||
| from nova import quota | ||||
| from nova import rpc | ||||
| from nova import utils | ||||
| from nova.api.ec2.cloud import ec2_id_to_id | ||||
| from nova.api.ec2 import ec2utils | ||||
| from nova.auth import manager | ||||
| from nova.cloudpipe import pipelib | ||||
| from nova.compute import instance_types | ||||
| @@ -94,6 +96,7 @@ flags.DECLARE('network_size', 'nova.network.manager') | ||||
| flags.DECLARE('vlan_start', 'nova.network.manager') | ||||
| flags.DECLARE('vpn_start', 'nova.network.manager') | ||||
| flags.DECLARE('fixed_range_v6', 'nova.network.manager') | ||||
| flags.DECLARE('images_path', 'nova.image.local') | ||||
| flags.DEFINE_flag(flags.HelpFlag()) | ||||
| flags.DEFINE_flag(flags.HelpshortFlag()) | ||||
| flags.DEFINE_flag(flags.HelpXMLFlag()) | ||||
| @@ -104,7 +107,7 @@ def param2id(object_id): | ||||
|     args: [object_id], e.g. 'vol-0000000a' or 'volume-0000000a' or '10' | ||||
|     """ | ||||
|     if '-' in object_id: | ||||
|         return ec2_id_to_id(object_id) | ||||
|         return ec2utils.ec2_id_to_id(object_id) | ||||
|     else: | ||||
|         return int(object_id) | ||||
|  | ||||
| @@ -545,6 +548,15 @@ class NetworkCommands(object): | ||||
|                                 network.dhcp_start, | ||||
|                                 network.dns) | ||||
|  | ||||
|     def delete(self, fixed_range): | ||||
|         """Deletes a network""" | ||||
|         network = db.network_get_by_cidr(context.get_admin_context(), \ | ||||
|                                          fixed_range) | ||||
|         if network.project_id is not None: | ||||
|             raise ValueError(_('Network must be disassociated from project %s' | ||||
|                                ' before delete' % network.project_id)) | ||||
|         db.network_delete_safe(context.get_admin_context(), network.id) | ||||
|  | ||||
|  | ||||
| class ServiceCommands(object): | ||||
|     """Enable and disable running services""" | ||||
| @@ -735,6 +747,155 @@ class InstanceTypeCommands(object): | ||||
|             self._print_instance_types(name, inst_types) | ||||
|  | ||||
|  | ||||
| class ImageCommands(object): | ||||
|     """Methods for dealing with a cloud in an odd state""" | ||||
|  | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         self.image_service = utils.import_object(FLAGS.image_service) | ||||
|  | ||||
|     def _register(self, image_type, disk_format, container_format, | ||||
|                   path, owner, name=None, is_public='T', | ||||
|                   architecture='x86_64', kernel_id=None, ramdisk_id=None): | ||||
|         meta = {'is_public': True, | ||||
|                 'name': name, | ||||
|                 'disk_format': disk_format, | ||||
|                 'container_format': container_format, | ||||
|                 'properties': {'image_state': 'available', | ||||
|                                'owner': owner, | ||||
|                                'type': image_type, | ||||
|                                'architecture': architecture, | ||||
|                                'image_location': 'local', | ||||
|                                'is_public': (is_public == 'T')}} | ||||
|         print image_type, meta | ||||
|         if kernel_id: | ||||
|             meta['properties']['kernel_id'] = int(kernel_id) | ||||
|         if ramdisk_id: | ||||
|             meta['properties']['ramdisk_id'] = int(ramdisk_id) | ||||
|         elevated = context.get_admin_context() | ||||
|         try: | ||||
|             with open(path) as ifile: | ||||
|                 image = self.image_service.create(elevated, meta, ifile) | ||||
|             new = image['id'] | ||||
|             print _("Image registered to %(new)s (%(new)08x).") % locals() | ||||
|             return new | ||||
|         except Exception as exc: | ||||
|             print _("Failed to register %(path)s: %(exc)s") % locals() | ||||
|  | ||||
|     def all_register(self, image, kernel, ramdisk, owner, name=None, | ||||
|                  is_public='T', architecture='x86_64'): | ||||
|         """Uploads an image, kernel, and ramdisk into the image_service | ||||
|         arguments: image kernel ramdisk owner [name] [is_public='T'] | ||||
|                    [architecture='x86_64']""" | ||||
|         kernel_id = self.kernel_register(kernel, owner, None, | ||||
|                                    is_public, architecture) | ||||
|         ramdisk_id = self.ramdisk_register(ramdisk, owner, None, | ||||
|                                     is_public, architecture) | ||||
|         self.image_register(image, owner, name, is_public, | ||||
|                        architecture, kernel_id, ramdisk_id) | ||||
|  | ||||
|     def image_register(self, path, owner, name=None, is_public='T', | ||||
|                        architecture='x86_64', kernel_id=None, ramdisk_id=None, | ||||
|                        disk_format='ami', container_format='ami'): | ||||
|         """Uploads an image into the image_service | ||||
|         arguments: path owner [name] [is_public='T'] [architecture='x86_64'] | ||||
|                    [kernel_id=None] [ramdisk_id=None] | ||||
|                    [disk_format='ami'] [container_format='ami']""" | ||||
|         return self._register('machine', disk_format, container_format, path, | ||||
|                               owner, name, is_public, architecture, | ||||
|                               kernel_id, ramdisk_id) | ||||
|  | ||||
|     def kernel_register(self, path, owner, name=None, is_public='T', | ||||
|                architecture='x86_64'): | ||||
|         """Uploads a kernel into the image_service | ||||
|         arguments: path owner [name] [is_public='T'] [architecture='x86_64'] | ||||
|         """ | ||||
|         return self._register('kernel', 'aki', 'aki', path, owner, name, | ||||
|                               is_public, architecture) | ||||
|  | ||||
|     def ramdisk_register(self, path, owner, name=None, is_public='T', | ||||
|                 architecture='x86_64'): | ||||
|         """Uploads a ramdisk into the image_service | ||||
|         arguments: path owner [name] [is_public='T'] [architecture='x86_64'] | ||||
|         """ | ||||
|         return self._register('ramdisk', 'ari', 'ari', path, owner, name, | ||||
|                               is_public, architecture) | ||||
|  | ||||
|     def _lookup(self, old_image_id): | ||||
|         try: | ||||
|             internal_id = ec2utils.ec2_id_to_id(old_image_id) | ||||
|             image = self.image_service.show(context, internal_id) | ||||
|         except exception.NotFound: | ||||
|             image = self.image_service.show_by_name(context, old_image_id) | ||||
|         return image['id'] | ||||
|  | ||||
|     def _old_to_new(self, old): | ||||
|         mapping = {'machine': 'ami', | ||||
|                    'kernel': 'aki', | ||||
|                    'ramdisk': 'ari'} | ||||
|         container_format = mapping[old['type']] | ||||
|         disk_format = container_format | ||||
|         new = {'disk_format': disk_format, | ||||
|                'container_format': container_format, | ||||
|                'is_public': True, | ||||
|                'name': old['imageId'], | ||||
|                'properties': {'image_state': old['imageState'], | ||||
|                               'owner': old['imageOwnerId'], | ||||
|                               'architecture': old['architecture'], | ||||
|                               'type': old['type'], | ||||
|                               'image_location': old['imageLocation'], | ||||
|                               'is_public': old['isPublic']}} | ||||
|         if old.get('kernelId'): | ||||
|             new['properties']['kernel_id'] = self._lookup(old['kernelId']) | ||||
|         if old.get('ramdiskId'): | ||||
|             new['properties']['ramdisk_id'] = self._lookup(old['ramdiskId']) | ||||
|         return new | ||||
|  | ||||
|     def _convert_images(self, images): | ||||
|         elevated = context.get_admin_context() | ||||
|         for image_path, image_metadata in images.iteritems(): | ||||
|             meta = self._old_to_new(image_metadata) | ||||
|             old = meta['name'] | ||||
|             try: | ||||
|                 with open(image_path) as ifile: | ||||
|                     image = self.image_service.create(elevated, meta, ifile) | ||||
|                 new = image['id'] | ||||
|                 print _("Image %(old)s converted to " \ | ||||
|                         "%(new)s (%(new)08x).") % locals() | ||||
|             except Exception as exc: | ||||
|                 print _("Failed to convert %(old)s: %(exc)s") % locals() | ||||
|  | ||||
|     def convert(self, directory): | ||||
|         """Uploads old objectstore images in directory to new service | ||||
|         arguments: directory""" | ||||
|         machine_images = {} | ||||
|         other_images = {} | ||||
|         directory = os.path.abspath(directory) | ||||
|         # NOTE(vish): If we're importing from the images path dir, attempt | ||||
|         #             to move the files out of the way before importing | ||||
|         #             so we aren't writing to the same directory. This | ||||
|         #             may fail if the dir was a mointpoint. | ||||
|         if (FLAGS.image_service == 'nova.image.local.LocalImageService' | ||||
|             and directory == os.path.abspath(FLAGS.images_path)): | ||||
|             new_dir = "%s_bak" % directory | ||||
|             os.move(directory, new_dir) | ||||
|             os.mkdir(directory) | ||||
|             directory = new_dir | ||||
|         for fn in glob.glob("%s/*/info.json" % directory): | ||||
|             try: | ||||
|                 image_path = os.path.join(fn.rpartition('/')[0], 'image') | ||||
|                 with open(fn) as metadata_file: | ||||
|                     image_metadata = json.load(metadata_file) | ||||
|                 if image_metadata['type'] == 'machine': | ||||
|                     machine_images[image_path] = image_metadata | ||||
|                 else: | ||||
|                     other_images[image_path] = image_metadata | ||||
|             except Exception as exc: | ||||
|                 print _("Failed to load %(fn)s.") % locals() | ||||
|         # NOTE(vish): do kernels and ramdisks first so images | ||||
|         self._convert_images(other_images) | ||||
|         self._convert_images(machine_images) | ||||
|  | ||||
|  | ||||
| CATEGORIES = [ | ||||
|     ('user', UserCommands), | ||||
|     ('project', ProjectCommands), | ||||
| @@ -749,6 +910,7 @@ CATEGORIES = [ | ||||
|     ('db', DbCommands), | ||||
|     ('volume', VolumeCommands), | ||||
|     ('instance_type', InstanceTypeCommands), | ||||
|     ('image', ImageCommands), | ||||
|     ('flavor', InstanceTypeCommands)] | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -48,7 +48,6 @@ class Exchange(object): | ||||
|         nm = self.name | ||||
|         LOG.debug(_('(%(nm)s) publish (key: %(routing_key)s)' | ||||
|                 ' %(message)s') % locals()) | ||||
|         routing_key = routing_key.split('.')[0] | ||||
|         if routing_key in self._routes: | ||||
|             for f in self._routes[routing_key]: | ||||
|                 LOG.debug(_('Publishing to route %s'), f) | ||||
|   | ||||
| @@ -321,6 +321,8 @@ DEFINE_integer('auth_token_ttl', 3600, 'Seconds for auth tokens to linger') | ||||
|  | ||||
| DEFINE_string('state_path', os.path.join(os.path.dirname(__file__), '../'), | ||||
|               "Top-level directory for maintaining nova's state") | ||||
| DEFINE_string('lock_path', os.path.join(os.path.dirname(__file__), '../'), | ||||
|               "Directory for lock files") | ||||
| DEFINE_string('logdir', None, 'output to a per-service log file in named ' | ||||
|                               'directory') | ||||
|  | ||||
| @@ -346,7 +348,7 @@ DEFINE_string('scheduler_manager', 'nova.scheduler.manager.SchedulerManager', | ||||
|               'Manager for scheduler') | ||||
|  | ||||
| # The service to use for image search and retrieval | ||||
| DEFINE_string('image_service', 'nova.image.s3.S3ImageService', | ||||
| DEFINE_string('image_service', 'nova.image.local.LocalImageService', | ||||
|               'The service to use for retrieving and searching for images.') | ||||
|  | ||||
| DEFINE_string('host', socket.gethostname(), | ||||
|   | ||||
| @@ -32,6 +32,7 @@ flags.DECLARE('fake_network', 'nova.network.manager') | ||||
| FLAGS.network_size = 8 | ||||
| FLAGS.num_networks = 2 | ||||
| FLAGS.fake_network = True | ||||
| FLAGS.image_service = 'nova.image.local.LocalImageService' | ||||
| flags.DECLARE('num_shelves', 'nova.volume.driver') | ||||
| flags.DECLARE('blades_per_shelf', 'nova.volume.driver') | ||||
| flags.DECLARE('iscsi_num_targets', 'nova.volume.driver') | ||||
|   | ||||
| @@ -38,6 +38,8 @@ from nova import test | ||||
| from nova.auth import manager | ||||
| from nova.compute import power_state | ||||
| from nova.api.ec2 import cloud | ||||
| from nova.api.ec2 import ec2utils | ||||
| from nova.image import local | ||||
| from nova.objectstore import image | ||||
|  | ||||
|  | ||||
| @@ -76,6 +78,12 @@ class CloudTestCase(test.TestCase): | ||||
|                                               project=self.project) | ||||
|         host = self.network.get_network_host(self.context.elevated()) | ||||
|  | ||||
|         def fake_show(meh, context, id): | ||||
|             return {'id': 1, 'properties': {'kernel_id': 1, 'ramdisk_id': 1}} | ||||
|  | ||||
|         self.stubs.Set(local.LocalImageService, 'show', fake_show) | ||||
|         self.stubs.Set(local.LocalImageService, 'show_by_name', fake_show) | ||||
|  | ||||
|     def tearDown(self): | ||||
|         network_ref = db.project_get_network(self.context, | ||||
|                                              self.project.id) | ||||
| @@ -122,7 +130,7 @@ class CloudTestCase(test.TestCase): | ||||
|         self.cloud.allocate_address(self.context) | ||||
|         inst = db.instance_create(self.context, {'host': self.compute.host}) | ||||
|         fixed = self.network.allocate_fixed_ip(self.context, inst['id']) | ||||
|         ec2_id = cloud.id_to_ec2_id(inst['id']) | ||||
|         ec2_id = ec2utils.id_to_ec2_id(inst['id']) | ||||
|         self.cloud.associate_address(self.context, | ||||
|                                      instance_id=ec2_id, | ||||
|                                      public_ip=address) | ||||
| @@ -158,12 +166,12 @@ class CloudTestCase(test.TestCase): | ||||
|         vol2 = db.volume_create(self.context, {}) | ||||
|         result = self.cloud.describe_volumes(self.context) | ||||
|         self.assertEqual(len(result['volumeSet']), 2) | ||||
|         volume_id = cloud.id_to_ec2_id(vol2['id'], 'vol-%08x') | ||||
|         volume_id = ec2utils.id_to_ec2_id(vol2['id'], 'vol-%08x') | ||||
|         result = self.cloud.describe_volumes(self.context, | ||||
|                                              volume_id=[volume_id]) | ||||
|         self.assertEqual(len(result['volumeSet']), 1) | ||||
|         self.assertEqual( | ||||
|                 cloud.ec2_id_to_id(result['volumeSet'][0]['volumeId']), | ||||
|                 ec2utils.ec2_id_to_id(result['volumeSet'][0]['volumeId']), | ||||
|                 vol2['id']) | ||||
|         db.volume_destroy(self.context, vol1['id']) | ||||
|         db.volume_destroy(self.context, vol2['id']) | ||||
| @@ -188,8 +196,10 @@ class CloudTestCase(test.TestCase): | ||||
|     def test_describe_instances(self): | ||||
|         """Makes sure describe_instances works and filters results.""" | ||||
|         inst1 = db.instance_create(self.context, {'reservation_id': 'a', | ||||
|                                                   'image_id': 1, | ||||
|                                                   'host': 'host1'}) | ||||
|         inst2 = db.instance_create(self.context, {'reservation_id': 'a', | ||||
|                                                   'image_id': 1, | ||||
|                                                   'host': 'host2'}) | ||||
|         comp1 = db.service_create(self.context, {'host': 'host1', | ||||
|                                                  'availability_zone': 'zone1', | ||||
| @@ -200,7 +210,7 @@ class CloudTestCase(test.TestCase): | ||||
|         result = self.cloud.describe_instances(self.context) | ||||
|         result = result['reservationSet'][0] | ||||
|         self.assertEqual(len(result['instancesSet']), 2) | ||||
|         instance_id = cloud.id_to_ec2_id(inst2['id']) | ||||
|         instance_id = ec2utils.id_to_ec2_id(inst2['id']) | ||||
|         result = self.cloud.describe_instances(self.context, | ||||
|                                              instance_id=[instance_id]) | ||||
|         result = result['reservationSet'][0] | ||||
| @@ -215,10 +225,9 @@ class CloudTestCase(test.TestCase): | ||||
|         db.service_destroy(self.context, comp2['id']) | ||||
|  | ||||
|     def test_console_output(self): | ||||
|         image_id = FLAGS.default_image | ||||
|         instance_type = FLAGS.default_instance_type | ||||
|         max_count = 1 | ||||
|         kwargs = {'image_id': image_id, | ||||
|         kwargs = {'image_id': 'ami-1', | ||||
|                   'instance_type': instance_type, | ||||
|                   'max_count': max_count} | ||||
|         rv = self.cloud.run_instances(self.context, **kwargs) | ||||
| @@ -234,8 +243,7 @@ class CloudTestCase(test.TestCase): | ||||
|         greenthread.sleep(0.3) | ||||
|  | ||||
|     def test_ajax_console(self): | ||||
|         image_id = FLAGS.default_image | ||||
|         kwargs = {'image_id': image_id} | ||||
|         kwargs = {'image_id': 'ami-1'} | ||||
|         rv = self.cloud.run_instances(self.context, **kwargs) | ||||
|         instance_id = rv['instancesSet'][0]['instanceId'] | ||||
|         greenthread.sleep(0.3) | ||||
| @@ -347,7 +355,7 @@ class CloudTestCase(test.TestCase): | ||||
|  | ||||
|     def test_update_of_instance_display_fields(self): | ||||
|         inst = db.instance_create(self.context, {}) | ||||
|         ec2_id = cloud.id_to_ec2_id(inst['id']) | ||||
|         ec2_id = ec2utils.id_to_ec2_id(inst['id']) | ||||
|         self.cloud.update_instance(self.context, ec2_id, | ||||
|                                    display_name='c00l 1m4g3') | ||||
|         inst = db.instance_get(self.context, inst['id']) | ||||
| @@ -365,7 +373,7 @@ class CloudTestCase(test.TestCase): | ||||
|     def test_update_of_volume_display_fields(self): | ||||
|         vol = db.volume_create(self.context, {}) | ||||
|         self.cloud.update_volume(self.context, | ||||
|                                  cloud.id_to_ec2_id(vol['id'], 'vol-%08x'), | ||||
|                                  ec2utils.id_to_ec2_id(vol['id'], 'vol-%08x'), | ||||
|                                  display_name='c00l v0lum3') | ||||
|         vol = db.volume_get(self.context, vol['id']) | ||||
|         self.assertEqual('c00l v0lum3', vol['display_name']) | ||||
| @@ -374,7 +382,7 @@ class CloudTestCase(test.TestCase): | ||||
|     def test_update_of_volume_wont_update_private_fields(self): | ||||
|         vol = db.volume_create(self.context, {}) | ||||
|         self.cloud.update_volume(self.context, | ||||
|                                  cloud.id_to_ec2_id(vol['id'], 'vol-%08x'), | ||||
|                                  ec2utils.id_to_ec2_id(vol['id'], 'vol-%08x'), | ||||
|                                  mountpoint='/not/here') | ||||
|         vol = db.volume_get(self.context, vol['id']) | ||||
|         self.assertEqual(None, vol['mountpoint']) | ||||
|   | ||||
| @@ -31,7 +31,7 @@ from nova import test | ||||
| from nova import utils | ||||
| from nova.auth import manager | ||||
| from nova.compute import instance_types | ||||
|  | ||||
| from nova.image import local | ||||
|  | ||||
| LOG = logging.getLogger('nova.tests.compute') | ||||
| FLAGS = flags.FLAGS | ||||
| @@ -52,6 +52,11 @@ class ComputeTestCase(test.TestCase): | ||||
|         self.project = self.manager.create_project('fake', 'fake', 'fake') | ||||
|         self.context = context.RequestContext('fake', 'fake', False) | ||||
|  | ||||
|         def fake_show(meh, context, id): | ||||
|             return {'id': 1, 'properties': {'kernel_id': 1, 'ramdisk_id': 1}} | ||||
|  | ||||
|         self.stubs.Set(local.LocalImageService, 'show', fake_show) | ||||
|  | ||||
|     def tearDown(self): | ||||
|         self.manager.delete_user(self.user) | ||||
|         self.manager.delete_project(self.project) | ||||
| @@ -60,7 +65,7 @@ class ComputeTestCase(test.TestCase): | ||||
|     def _create_instance(self, params={}): | ||||
|         """Create a test instance""" | ||||
|         inst = {} | ||||
|         inst['image_id'] = 'ami-test' | ||||
|         inst['image_id'] = 1 | ||||
|         inst['reservation_id'] = 'r-fakeres' | ||||
|         inst['launch_time'] = '10' | ||||
|         inst['user_id'] = self.user.id | ||||
|   | ||||
| @@ -57,7 +57,7 @@ class ConsoleTestCase(test.TestCase): | ||||
|         inst = {} | ||||
|         #inst['host'] = self.host | ||||
|         #inst['name'] = 'instance-1234' | ||||
|         inst['image_id'] = 'ami-test' | ||||
|         inst['image_id'] = 1 | ||||
|         inst['reservation_id'] = 'r-fakeres' | ||||
|         inst['launch_time'] = '10' | ||||
|         inst['user_id'] = self.user.id | ||||
|   | ||||
| @@ -59,6 +59,7 @@ class DirectTestCase(test.TestCase): | ||||
|         req.headers['X-OpenStack-User'] = 'user1' | ||||
|         req.headers['X-OpenStack-Project'] = 'proj1' | ||||
|         resp = req.get_response(self.auth_router) | ||||
|         self.assertEqual(resp.status_int, 200) | ||||
|         data = json.loads(resp.body) | ||||
|         self.assertEqual(data['user'], 'user1') | ||||
|         self.assertEqual(data['project'], 'proj1') | ||||
| @@ -69,6 +70,7 @@ class DirectTestCase(test.TestCase): | ||||
|         req.method = 'POST' | ||||
|         req.body = 'json=%s' % json.dumps({'data': 'foo'}) | ||||
|         resp = req.get_response(self.router) | ||||
|         self.assertEqual(resp.status_int, 200) | ||||
|         resp_parsed = json.loads(resp.body) | ||||
|         self.assertEqual(resp_parsed['data'], 'foo') | ||||
|  | ||||
| @@ -78,6 +80,7 @@ class DirectTestCase(test.TestCase): | ||||
|         req.method = 'POST' | ||||
|         req.body = 'data=foo' | ||||
|         resp = req.get_response(self.router) | ||||
|         self.assertEqual(resp.status_int, 200) | ||||
|         resp_parsed = json.loads(resp.body) | ||||
|         self.assertEqual(resp_parsed['data'], 'foo') | ||||
|  | ||||
| @@ -90,8 +93,7 @@ class DirectTestCase(test.TestCase): | ||||
| class DirectCloudTestCase(test_cloud.CloudTestCase): | ||||
|     def setUp(self): | ||||
|         super(DirectCloudTestCase, self).setUp() | ||||
|         compute_handle = compute.API(image_service=self.cloud.image_service, | ||||
|                                      network_api=self.cloud.network_api, | ||||
|         compute_handle = compute.API(network_api=self.cloud.network_api, | ||||
|                                      volume_api=self.cloud.volume_api) | ||||
|         direct.register_service('compute', compute_handle) | ||||
|         self.router = direct.JsonParamsMiddleware(direct.Router()) | ||||
|   | ||||
| @@ -14,10 +14,12 @@ | ||||
| #    License for the specific language governing permissions and limitations | ||||
| #    under the License. | ||||
|  | ||||
| import errno | ||||
| import os | ||||
| import select | ||||
|  | ||||
| from nova import test | ||||
| from nova.utils import parse_mailmap, str_dict_replace | ||||
| from nova.utils import parse_mailmap, str_dict_replace, synchronized | ||||
|  | ||||
|  | ||||
| class ProjectTestCase(test.TestCase): | ||||
| @@ -55,3 +57,47 @@ class ProjectTestCase(test.TestCase): | ||||
|                                 '%r not listed in Authors' % missing) | ||||
|             finally: | ||||
|                 tree.unlock() | ||||
|  | ||||
|  | ||||
| class LockTestCase(test.TestCase): | ||||
|     def test_synchronized_wrapped_function_metadata(self): | ||||
|         @synchronized('whatever') | ||||
|         def foo(): | ||||
|             """Bar""" | ||||
|             pass | ||||
|         self.assertEquals(foo.__doc__, 'Bar', "Wrapped function's docstring " | ||||
|                                               "got lost") | ||||
|         self.assertEquals(foo.__name__, 'foo', "Wrapped function's name " | ||||
|                                                "got mangled") | ||||
|  | ||||
|     def test_synchronized(self): | ||||
|         rpipe1, wpipe1 = os.pipe() | ||||
|         rpipe2, wpipe2 = os.pipe() | ||||
|  | ||||
|         @synchronized('testlock') | ||||
|         def f(rpipe, wpipe): | ||||
|             try: | ||||
|                 os.write(wpipe, "foo") | ||||
|             except OSError, e: | ||||
|                 self.assertEquals(e.errno, errno.EPIPE) | ||||
|                 return | ||||
|  | ||||
|             rfds, _, __ = select.select([rpipe], [], [], 1) | ||||
|             self.assertEquals(len(rfds), 0, "The other process, which was" | ||||
|                                             " supposed to be locked, " | ||||
|                                             "wrote on its end of the " | ||||
|                                             "pipe") | ||||
|             os.close(rpipe) | ||||
|  | ||||
|         pid = os.fork() | ||||
|         if pid > 0: | ||||
|             os.close(wpipe1) | ||||
|             os.close(rpipe2) | ||||
|  | ||||
|             f(rpipe1, wpipe2) | ||||
|         else: | ||||
|             os.close(rpipe1) | ||||
|             os.close(wpipe2) | ||||
|  | ||||
|             f(rpipe2, wpipe1) | ||||
|             os._exit(0) | ||||
|   | ||||
| @@ -343,13 +343,13 @@ def lease_ip(private_ip): | ||||
|                                           private_ip) | ||||
|     instance_ref = db.fixed_ip_get_instance(context.get_admin_context(), | ||||
|                                             private_ip) | ||||
|     cmd = "%s add %s %s fake" % (binpath('nova-dhcpbridge'), | ||||
|     cmd = (binpath('nova-dhcpbridge'), 'add', | ||||
|            instance_ref['mac_address'], | ||||
|                                  private_ip) | ||||
|            private_ip, 'fake') | ||||
|     env = {'DNSMASQ_INTERFACE': network_ref['bridge'], | ||||
|            'TESTING': '1', | ||||
|            'FLAGFILE': FLAGS.dhcpbridge_flagfile} | ||||
|     (out, err) = utils.execute(cmd, addl_env=env) | ||||
|     (out, err) = utils.execute(*cmd, addl_env=env) | ||||
|     LOG.debug("ISSUE_IP: %s, %s ", out, err) | ||||
|  | ||||
|  | ||||
| @@ -359,11 +359,11 @@ def release_ip(private_ip): | ||||
|                                           private_ip) | ||||
|     instance_ref = db.fixed_ip_get_instance(context.get_admin_context(), | ||||
|                                             private_ip) | ||||
|     cmd = "%s del %s %s fake" % (binpath('nova-dhcpbridge'), | ||||
|     cmd = (binpath('nova-dhcpbridge'), 'del', | ||||
|            instance_ref['mac_address'], | ||||
|                                  private_ip) | ||||
|            private_ip, 'fake') | ||||
|     env = {'DNSMASQ_INTERFACE': network_ref['bridge'], | ||||
|            'TESTING': '1', | ||||
|            'FLAGFILE': FLAGS.dhcpbridge_flagfile} | ||||
|     (out, err) = utils.execute(cmd, addl_env=env) | ||||
|     (out, err) = utils.execute(*cmd, addl_env=env) | ||||
|     LOG.debug("RELEASE_IP: %s, %s ", out, err) | ||||
|   | ||||
| @@ -155,7 +155,7 @@ class SimpleDriverTestCase(test.TestCase): | ||||
|     def _create_instance(self, **kwargs): | ||||
|         """Create a test instance""" | ||||
|         inst = {} | ||||
|         inst['image_id'] = 'ami-test' | ||||
|         inst['image_id'] = 1 | ||||
|         inst['reservation_id'] = 'r-fakeres' | ||||
|         inst['user_id'] = self.user.id | ||||
|         inst['project_id'] = self.project.id | ||||
| @@ -169,8 +169,6 @@ class SimpleDriverTestCase(test.TestCase): | ||||
|     def _create_volume(self): | ||||
|         """Create a test volume""" | ||||
|         vol = {} | ||||
|         vol['image_id'] = 'ami-test' | ||||
|         vol['reservation_id'] = 'r-fakeres' | ||||
|         vol['size'] = 1 | ||||
|         vol['availability_zone'] = 'test' | ||||
|         return db.volume_create(self.context, vol)['id'] | ||||
|   | ||||
| @@ -14,6 +14,9 @@ | ||||
| #    License for the specific language governing permissions and limitations | ||||
| #    under the License. | ||||
|  | ||||
| import os | ||||
|  | ||||
| import eventlet | ||||
| from xml.etree.ElementTree import fromstring as xml_to_tree | ||||
| from xml.dom.minidom import parseString as xml_to_dom | ||||
|  | ||||
| @@ -30,6 +33,70 @@ FLAGS = flags.FLAGS | ||||
| flags.DECLARE('instances_path', 'nova.compute.manager') | ||||
|  | ||||
|  | ||||
| def _concurrency(wait, done, target): | ||||
|     wait.wait() | ||||
|     done.send() | ||||
|  | ||||
|  | ||||
| class CacheConcurrencyTestCase(test.TestCase): | ||||
|     def setUp(self): | ||||
|         super(CacheConcurrencyTestCase, self).setUp() | ||||
|  | ||||
|         def fake_exists(fname): | ||||
|             basedir = os.path.join(FLAGS.instances_path, '_base') | ||||
|             if fname == basedir: | ||||
|                 return True | ||||
|             return False | ||||
|  | ||||
|         def fake_execute(*args, **kwargs): | ||||
|             pass | ||||
|  | ||||
|         self.stubs.Set(os.path, 'exists', fake_exists) | ||||
|         self.stubs.Set(utils, 'execute', fake_execute) | ||||
|  | ||||
|     def test_same_fname_concurrency(self): | ||||
|         """Ensures that the same fname cache runs at a sequentially""" | ||||
|         conn = libvirt_conn.LibvirtConnection | ||||
|         wait1 = eventlet.event.Event() | ||||
|         done1 = eventlet.event.Event() | ||||
|         eventlet.spawn(conn._cache_image, _concurrency, | ||||
|                        'target', 'fname', False, wait1, done1) | ||||
|         wait2 = eventlet.event.Event() | ||||
|         done2 = eventlet.event.Event() | ||||
|         eventlet.spawn(conn._cache_image, _concurrency, | ||||
|                        'target', 'fname', False, wait2, done2) | ||||
|         wait2.send() | ||||
|         eventlet.sleep(0) | ||||
|         try: | ||||
|             self.assertFalse(done2.ready()) | ||||
|             self.assertTrue('fname' in conn._image_sems) | ||||
|         finally: | ||||
|             wait1.send() | ||||
|         done1.wait() | ||||
|         eventlet.sleep(0) | ||||
|         self.assertTrue(done2.ready()) | ||||
|         self.assertFalse('fname' in conn._image_sems) | ||||
|  | ||||
|     def test_different_fname_concurrency(self): | ||||
|         """Ensures that two different fname caches are concurrent""" | ||||
|         conn = libvirt_conn.LibvirtConnection | ||||
|         wait1 = eventlet.event.Event() | ||||
|         done1 = eventlet.event.Event() | ||||
|         eventlet.spawn(conn._cache_image, _concurrency, | ||||
|                        'target', 'fname2', False, wait1, done1) | ||||
|         wait2 = eventlet.event.Event() | ||||
|         done2 = eventlet.event.Event() | ||||
|         eventlet.spawn(conn._cache_image, _concurrency, | ||||
|                        'target', 'fname1', False, wait2, done2) | ||||
|         wait2.send() | ||||
|         eventlet.sleep(0) | ||||
|         try: | ||||
|             self.assertTrue(done2.ready()) | ||||
|         finally: | ||||
|             wait1.send() | ||||
|             eventlet.sleep(0) | ||||
|  | ||||
|  | ||||
| class LibvirtConnTestCase(test.TestCase): | ||||
|     def setUp(self): | ||||
|         super(LibvirtConnTestCase, self).setUp() | ||||
| @@ -315,15 +382,16 @@ class IptablesFirewallTestCase(test.TestCase): | ||||
|         instance_ref = db.instance_get(admin_ctxt, instance_ref['id']) | ||||
|  | ||||
| #        self.fw.add_instance(instance_ref) | ||||
|         def fake_iptables_execute(cmd, process_input=None): | ||||
|             if cmd == 'sudo ip6tables-save -t filter': | ||||
|         def fake_iptables_execute(*cmd, **kwargs): | ||||
|             process_input = kwargs.get('process_input', None) | ||||
|             if cmd == ('sudo', 'ip6tables-save', '-t', 'filter'): | ||||
|                 return '\n'.join(self.in6_rules), None | ||||
|             if cmd == 'sudo iptables-save -t filter': | ||||
|             if cmd == ('sudo', 'iptables-save', '-t', 'filter'): | ||||
|                 return '\n'.join(self.in_rules), None | ||||
|             if cmd == 'sudo iptables-restore': | ||||
|             if cmd == ('sudo', 'iptables-restore'): | ||||
|                 self.out_rules = process_input.split('\n') | ||||
|                 return '', '' | ||||
|             if cmd == 'sudo ip6tables-restore': | ||||
|             if cmd == ('sudo', 'ip6tables-restore'): | ||||
|                 self.out6_rules = process_input.split('\n') | ||||
|                 return '', '' | ||||
|         self.fw.execute = fake_iptables_execute | ||||
|   | ||||
| @@ -99,7 +99,7 @@ class VolumeTestCase(test.TestCase): | ||||
|     def test_run_attach_detach_volume(self): | ||||
|         """Make sure volume can be attached and detached from instance.""" | ||||
|         inst = {} | ||||
|         inst['image_id'] = 'ami-test' | ||||
|         inst['image_id'] = 1 | ||||
|         inst['reservation_id'] = 'r-fakeres' | ||||
|         inst['launch_time'] = '10' | ||||
|         inst['user_id'] = 'fake' | ||||
|   | ||||
| @@ -474,16 +474,28 @@ class XenAPIMigrateInstance(test.TestCase): | ||||
|         db_fakes.stub_out_db_instance_api(self.stubs) | ||||
|         stubs.stub_out_get_target(self.stubs) | ||||
|         xenapi_fake.reset() | ||||
|         self.manager = manager.AuthManager() | ||||
|         self.user = self.manager.create_user('fake', 'fake', 'fake', | ||||
|                                              admin=True) | ||||
|         self.project = self.manager.create_project('fake', 'fake', 'fake') | ||||
|         self.values = {'name': 1, 'id': 1, | ||||
|                   'project_id': 'fake', | ||||
|                   'user_id': 'fake', | ||||
|                   'project_id': self.project.id, | ||||
|                   'user_id': self.user.id, | ||||
|                   'image_id': 1, | ||||
|                   'kernel_id': 2, | ||||
|                   'ramdisk_id': 3, | ||||
|                   'kernel_id': None, | ||||
|                   'ramdisk_id': None, | ||||
|                   'instance_type': 'm1.large', | ||||
|                   'mac_address': 'aa:bb:cc:dd:ee:ff', | ||||
|                   } | ||||
|         stubs.stub_out_migration_methods(self.stubs) | ||||
|         glance_stubs.stubout_glance_client(self.stubs, | ||||
|                                            glance_stubs.FakeGlance) | ||||
|  | ||||
|     def tearDown(self): | ||||
|         super(XenAPIMigrateInstance, self).tearDown() | ||||
|         self.manager.delete_project(self.project) | ||||
|         self.manager.delete_user(self.user) | ||||
|         self.stubs.UnsetAll() | ||||
|  | ||||
|     def test_migrate_disk_and_power_off(self): | ||||
|         instance = db.instance_create(self.values) | ||||
| @@ -491,11 +503,11 @@ class XenAPIMigrateInstance(test.TestCase): | ||||
|         conn = xenapi_conn.get_connection(False) | ||||
|         conn.migrate_disk_and_power_off(instance, '127.0.0.1') | ||||
|  | ||||
|     def test_attach_disk(self): | ||||
|     def test_finish_resize(self): | ||||
|         instance = db.instance_create(self.values) | ||||
|         stubs.stubout_session(self.stubs, stubs.FakeSessionForMigrationTests) | ||||
|         conn = xenapi_conn.get_connection(False) | ||||
|         conn.attach_disk(instance, {'base_copy': 'hurr', 'cow': 'durr'}) | ||||
|         conn.finish_resize(instance, dict(base_copy='hurr', cow='durr')) | ||||
|  | ||||
|  | ||||
| class XenAPIDetermineDiskImageTestCase(test.TestCase): | ||||
|   | ||||
| @@ -231,6 +231,18 @@ class FakeSessionForMigrationTests(fake.SessionBase): | ||||
|     def __init__(self, uri): | ||||
|         super(FakeSessionForMigrationTests, self).__init__(uri) | ||||
|  | ||||
|     def VDI_get_by_uuid(*args): | ||||
|         return 'hurr' | ||||
|  | ||||
|     def VM_start(self, _1, ref, _2, _3): | ||||
|         vm = fake.get_record('VM', ref) | ||||
|         if vm['power_state'] != 'Halted': | ||||
|             raise fake.Failure(['VM_BAD_POWER_STATE', ref, 'Halted', | ||||
|                                   vm['power_state']]) | ||||
|         vm['power_state'] = 'Running' | ||||
|         vm['is_a_template'] = False | ||||
|         vm['is_control_domain'] = False | ||||
|  | ||||
|  | ||||
| def stub_out_migration_methods(stubs): | ||||
|     def fake_get_snapshot(self, instance): | ||||
| @@ -257,6 +269,9 @@ def stub_out_migration_methods(stubs): | ||||
|     def fake_destroy(*args, **kwargs): | ||||
|         pass | ||||
|  | ||||
|     def fake_reset_network(*args, **kwargs): | ||||
|         pass | ||||
|  | ||||
|     stubs.Set(vmops.VMOps, '_destroy', fake_destroy) | ||||
|     stubs.Set(vm_utils.VMHelper, 'scan_default_sr', fake_sr) | ||||
|     stubs.Set(vm_utils.VMHelper, 'scan_sr', fake_sr) | ||||
| @@ -264,4 +279,5 @@ def stub_out_migration_methods(stubs): | ||||
|     stubs.Set(vm_utils.VMHelper, 'get_vdi_for_vm_safely', fake_get_vdi) | ||||
|     stubs.Set(xenapi_conn.XenAPISession, 'wait_for_task', lambda x, y, z: None) | ||||
|     stubs.Set(vm_utils.VMHelper, 'get_sr_path', fake_get_sr_path) | ||||
|     stubs.Set(vmops.VMOps, 'reset_network', fake_reset_network) | ||||
|     stubs.Set(vmops.VMOps, '_shutdown', fake_shutdown) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Salvatore Orlando
					Salvatore Orlando