Fix server create for boot-from-volume

* server create required --image even when booting the server from a
  volume.  Change options to require either --image or --volume to
  specify the server boot disk.  Using --volume currently uses device
  'vda' for the block mapping and ignores any other block mappings
  given in --block-device-mapping.
* server create and server show are both affected by bug 1378842 where
  an excepion was thrown when no image ID was present in the returned
  server object, which is the case for a server booted from a volume.
* Fix the remaining assertEqual() order problems in test_server.py

Closes-Bug: 1378842
Closes-Bug: 1383338
Change-Id: I5daebf4e50a765d4920088dfead95b6295af6a4d
This commit is contained in:
Dean Troyer 2014-10-20 11:43:29 -05:00
parent 8ba74451ee
commit 2c9d263611
2 changed files with 52 additions and 21 deletions

View File

@ -67,6 +67,7 @@ def _prep_server_detail(compute_client, server):
# Convert the image blob to a name # Convert the image blob to a name
image_info = info.get('image', {}) image_info = info.get('image', {})
if image_info:
image_id = image_info.get('id', '') image_id = image_info.get('id', '')
image = utils.find_resource(compute_client.images, image_id) image = utils.find_resource(compute_client.images, image_id)
info['image'] = "%s (%s)" % (image.name, image_id) info['image'] = "%s (%s)" % (image.name, image_id)
@ -192,11 +193,17 @@ class CreateServer(show.ShowOne):
'server_name', 'server_name',
metavar='<server-name>', metavar='<server-name>',
help=_('New server name')) help=_('New server name'))
parser.add_argument( disk_group = parser.add_mutually_exclusive_group(
required=True,
)
disk_group.add_argument(
'--image', '--image',
metavar='<image>', metavar='<image>',
required=True,
help=_('Create server from this image')) help=_('Create server from this image'))
disk_group.add_argument(
'--volume',
metavar='<volume>',
help=_('Create server from this volume'))
parser.add_argument( parser.add_argument(
'--flavor', '--flavor',
metavar='<flavor>', metavar='<flavor>',
@ -282,10 +289,23 @@ class CreateServer(show.ShowOne):
def take_action(self, parsed_args): def take_action(self, parsed_args):
self.log.debug('take_action(%s)', parsed_args) self.log.debug('take_action(%s)', parsed_args)
compute_client = self.app.client_manager.compute compute_client = self.app.client_manager.compute
volume_client = self.app.client_manager.volume
# Lookup parsed_args.image # Lookup parsed_args.image
image = utils.find_resource(compute_client.images, image = None
parsed_args.image) if parsed_args.image:
image = utils.find_resource(
compute_client.images,
parsed_args.image,
)
# Lookup parsed_args.volume
volume = None
if parsed_args.volume:
volume = utils.find_resource(
volume_client.volumes,
parsed_args.volume,
).id
# Lookup parsed_args.flavor # Lookup parsed_args.flavor
flavor = utils.find_resource(compute_client.flavors, flavor = utils.find_resource(compute_client.flavors,
@ -319,8 +339,21 @@ class CreateServer(show.ShowOne):
msg = "Can't open '%s': %s" msg = "Can't open '%s': %s"
raise exceptions.CommandError(msg % (parsed_args.user_data, e)) raise exceptions.CommandError(msg % (parsed_args.user_data, e))
block_device_mapping = dict(v.split('=', 1) block_device_mapping = {}
for v in parsed_args.block_device_mapping) if volume:
# When booting from volume, for now assume no other mappings
# This device value is likely KVM-specific
block_device_mapping = {'vda': volume}
else:
for dev_map in parsed_args.block_device_mapping:
dev_key, dev_vol = dev_map.split('=', 1)
block_volume = None
if dev_vol:
block_volume = utils.find_resource(
volume_client.volumes,
dev_vol,
).id
block_device_mapping.update({dev_key: block_volume})
nics = [] nics = []
for nic_str in parsed_args.nic: for nic_str in parsed_args.nic:

View File

@ -134,17 +134,16 @@ class TestServerCreate(TestServer):
**kwargs **kwargs
) )
collist = ('addresses', 'flavor', 'id', 'image', 'name', 'properties') collist = ('addresses', 'flavor', 'id', 'name', 'properties')
self.assertEqual(columns, collist) self.assertEqual(collist, columns)
datalist = ( datalist = (
'', '',
'Large ()', 'Large ()',
compute_fakes.server_id, compute_fakes.server_id,
'graven ()',
compute_fakes.server_name, compute_fakes.server_name,
'', '',
) )
self.assertEqual(data, datalist) self.assertEqual(datalist, data)
@mock.patch('openstackclient.compute.v2.server.io.open') @mock.patch('openstackclient.compute.v2.server.io.open')
def test_server_create_userdata(self, mock_open): def test_server_create_userdata(self, mock_open):
@ -200,17 +199,16 @@ class TestServerCreate(TestServer):
**kwargs **kwargs
) )
collist = ('addresses', 'flavor', 'id', 'image', 'name', 'properties') collist = ('addresses', 'flavor', 'id', 'name', 'properties')
self.assertEqual(columns, collist) self.assertEqual(collist, columns)
datalist = ( datalist = (
'', '',
'Large ()', 'Large ()',
compute_fakes.server_id, compute_fakes.server_id,
'graven ()',
compute_fakes.server_name, compute_fakes.server_name,
'', '',
) )
self.assertEqual(data, datalist) self.assertEqual(datalist, data)
class TestServerDelete(TestServer): class TestServerDelete(TestServer):
@ -288,14 +286,14 @@ class TestServerImageCreate(TestServer):
) )
collist = ('id', 'is_public', 'name', 'owner') collist = ('id', 'is_public', 'name', 'owner')
self.assertEqual(columns, collist) self.assertEqual(collist, columns)
datalist = ( datalist = (
image_fakes.image_id, image_fakes.image_id,
False, False,
image_fakes.image_name, image_fakes.image_name,
image_fakes.image_owner, image_fakes.image_owner,
) )
self.assertEqual(data, datalist) self.assertEqual(datalist, data)
def test_server_image_create_name(self): def test_server_image_create_name(self):
arglist = [ arglist = [
@ -318,14 +316,14 @@ class TestServerImageCreate(TestServer):
) )
collist = ('id', 'is_public', 'name', 'owner') collist = ('id', 'is_public', 'name', 'owner')
self.assertEqual(columns, collist) self.assertEqual(collist, columns)
datalist = ( datalist = (
image_fakes.image_id, image_fakes.image_id,
False, False,
image_fakes.image_name, image_fakes.image_name,
image_fakes.image_owner, image_fakes.image_owner,
) )
self.assertEqual(data, datalist) self.assertEqual(datalist, data)
class TestServerResize(TestServer): class TestServerResize(TestServer):