
* Rewrite of Volume code to make VolumeManager more generic * AoE vs. iscsi moved to driver layer * Added db support for target ids * Added code to re-export volumes on restart of VolumeManager * Includes a script to create /dev/iscsi volumes on remote hosts
181 lines
7.1 KiB
Python
181 lines
7.1 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.
|
|
# 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 Volume Code
|
|
"""
|
|
import logging
|
|
|
|
from twisted.internet import defer
|
|
|
|
from nova import context
|
|
from nova import exception
|
|
from nova import db
|
|
from nova import flags
|
|
from nova import test
|
|
from nova import utils
|
|
|
|
FLAGS = flags.FLAGS
|
|
|
|
|
|
class VolumeTestCase(test.TrialTestCase):
|
|
"""Test Case for volumes"""
|
|
def setUp(self):
|
|
logging.getLogger().setLevel(logging.DEBUG)
|
|
super(VolumeTestCase, self).setUp()
|
|
self.compute = utils.import_object(FLAGS.compute_manager)
|
|
self.flags(connection_type='fake')
|
|
self.volume = utils.import_object(FLAGS.volume_manager)
|
|
self.context = context.get_admin_context()
|
|
|
|
@staticmethod
|
|
def _create_volume(size='0'):
|
|
"""Create a volume object"""
|
|
vol = {}
|
|
vol['size'] = size
|
|
vol['user_id'] = 'fake'
|
|
vol['project_id'] = 'fake'
|
|
vol['availability_zone'] = FLAGS.storage_availability_zone
|
|
vol['status'] = "creating"
|
|
vol['attach_status'] = "detached"
|
|
return db.volume_create(context.get_admin_context(), vol)['id']
|
|
|
|
@defer.inlineCallbacks
|
|
def test_create_delete_volume(self):
|
|
"""Test volume can be created and deleted"""
|
|
volume_id = self._create_volume()
|
|
yield self.volume.create_volume(self.context, volume_id)
|
|
self.assertEqual(volume_id, db.volume_get(context.get_admin_context(),
|
|
volume_id).id)
|
|
|
|
yield self.volume.delete_volume(self.context, volume_id)
|
|
self.assertRaises(exception.NotFound,
|
|
db.volume_get,
|
|
self.context,
|
|
volume_id)
|
|
|
|
@defer.inlineCallbacks
|
|
def test_too_big_volume(self):
|
|
"""Ensure failure if a too large of a volume is requested"""
|
|
# FIXME(vish): validation needs to move into the data layer in
|
|
# volume_create
|
|
defer.returnValue(True)
|
|
try:
|
|
volume_id = self._create_volume('1001')
|
|
yield self.volume.create_volume(self.context, volume_id)
|
|
self.fail("Should have thrown TypeError")
|
|
except TypeError:
|
|
pass
|
|
|
|
@defer.inlineCallbacks
|
|
def test_too_many_volumes(self):
|
|
"""Ensure that NoMoreTargets is raised when we run out of volumes"""
|
|
vols = []
|
|
total_slots = FLAGS.iscsi_target_ids
|
|
for _index in xrange(total_slots):
|
|
volume_id = self._create_volume()
|
|
yield self.volume.create_volume(self.context, volume_id)
|
|
vols.append(volume_id)
|
|
volume_id = self._create_volume()
|
|
self.assertFailure(self.volume.create_volume(self.context,
|
|
volume_id),
|
|
db.NoMoreTargets)
|
|
db.volume_destroy(context.get_admin_context(), volume_id)
|
|
for volume_id in vols:
|
|
yield self.volume.delete_volume(self.context, volume_id)
|
|
|
|
@defer.inlineCallbacks
|
|
def test_run_attach_detach_volume(self):
|
|
"""Make sure volume can be attached and detached from instance"""
|
|
inst = {}
|
|
inst['image_id'] = 'ami-test'
|
|
inst['reservation_id'] = 'r-fakeres'
|
|
inst['launch_time'] = '10'
|
|
inst['user_id'] = 'fake'
|
|
inst['project_id'] = 'fake'
|
|
inst['instance_type'] = 'm1.tiny'
|
|
inst['mac_address'] = utils.generate_mac()
|
|
inst['ami_launch_index'] = 0
|
|
instance_id = db.instance_create(self.context, inst)['id']
|
|
mountpoint = "/dev/sdf"
|
|
volume_id = self._create_volume()
|
|
yield self.volume.create_volume(self.context, volume_id)
|
|
if FLAGS.fake_tests:
|
|
db.volume_attached(self.context, volume_id, instance_id,
|
|
mountpoint)
|
|
else:
|
|
yield self.compute.attach_volume(self.context,
|
|
instance_id,
|
|
volume_id,
|
|
mountpoint)
|
|
vol = db.volume_get(context.get_admin_context(), volume_id)
|
|
self.assertEqual(vol['status'], "in-use")
|
|
self.assertEqual(vol['attach_status'], "attached")
|
|
self.assertEqual(vol['mountpoint'], mountpoint)
|
|
instance_ref = db.volume_get_instance(self.context, volume_id)
|
|
self.assertEqual(instance_ref['id'], instance_id)
|
|
|
|
self.assertFailure(self.volume.delete_volume(self.context, volume_id),
|
|
exception.Error)
|
|
if FLAGS.fake_tests:
|
|
db.volume_detached(self.context, volume_id)
|
|
else:
|
|
yield self.compute.detach_volume(self.context,
|
|
instance_id,
|
|
volume_id)
|
|
vol = db.volume_get(self.context, volume_id)
|
|
self.assertEqual(vol['status'], "available")
|
|
|
|
yield self.volume.delete_volume(self.context, volume_id)
|
|
self.assertRaises(exception.Error,
|
|
db.volume_get,
|
|
self.context,
|
|
volume_id)
|
|
db.instance_destroy(self.context, instance_id)
|
|
|
|
@defer.inlineCallbacks
|
|
def test_concurrent_volumes_get_different_targets(self):
|
|
"""Ensure multiple concurrent volumes get different targets"""
|
|
volume_ids = []
|
|
targets = []
|
|
|
|
def _check(volume_id):
|
|
"""Make sure targets aren't duplicated"""
|
|
volume_ids.append(volume_id)
|
|
admin_context = context.get_admin_context()
|
|
target_id = db.volume_get_target_id(admin_context, volume_id)
|
|
self.assert_(target_id not in targets)
|
|
targets.append(target_id)
|
|
logging.debug("Target %s allocated", target_id)
|
|
deferreds = []
|
|
total_slots = FLAGS.iscsi_target_ids
|
|
for _index in xrange(total_slots):
|
|
volume_id = self._create_volume()
|
|
d = self.volume.create_volume(self.context, volume_id)
|
|
d.addCallback(_check)
|
|
d.addErrback(self.fail)
|
|
deferreds.append(d)
|
|
yield defer.DeferredList(deferreds)
|
|
for volume_id in volume_ids:
|
|
self.volume.delete_volume(self.context, volume_id)
|
|
|
|
def test_multi_node(self):
|
|
# TODO(termie): Figure out how to test with two nodes,
|
|
# each of them having a different FLAG for storage_node
|
|
# This will allow us to test cross-node interactions
|
|
pass
|