**** CubicPower OpenStack Study ****
# Copyright 2013 OpenStack Foundation
# 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.
"""
Functional tests for the RBD store interface.
Set the GLANCE_TEST_RBD_CONF environment variable to the location
of a Glance config that defines how to connect to a functional
RBD backend. This backend must be running Ceph Bobtail (0.56) or later.
"""
import ConfigParser
import os
import uuid
import oslo.config.cfg
import six
import testtools
from glance.common import exception
import glance.store.rbd
import glance.tests.functional.store as store_tests
try:
import rados
import rbd
except ImportError:
rados = None
**** CubicPower OpenStack Study ****
def read_config(path):
cp = ConfigParser.RawConfigParser()
cp.read(path)
return cp
**** CubicPower OpenStack Study ****
def parse_config(config):
out = {}
options = [
'rbd_store_chunk_size',
'rbd_store_pool',
'rbd_store_user',
'rbd_store_ceph_conf',
]
for option in options:
out[option] = config.defaults()[option]
return out
**** CubicPower OpenStack Study ****
class TestRBDStore(store_tests.BaseTestCase, testtools.TestCase):
store_cls_path = 'glance.store.rbd.Store'
store_cls = glance.store.rbd.Store
store_name = 'rbd'
**** CubicPower OpenStack Study ****
def setUp(self):
config_path = os.environ.get('GLANCE_TEST_RBD_CONF')
if not config_path:
msg = "GLANCE_TEST_RBD_CONF environ not set."
self.skipTest(msg)
oslo.config.cfg.CONF(args=[], default_config_files=[config_path])
raw_config = read_config(config_path)
config = parse_config(raw_config)
if rados is None:
self.skipTest("rados python library not found")
rados_client = rados.Rados(conffile=config['rbd_store_ceph_conf'],
rados_id=config['rbd_store_user'])
try:
rados_client.connect()
except rados.Error as e:
self.skipTest("Failed to connect to RADOS: %s" % e)
try:
rados_client.create_pool(config['rbd_store_pool'])
except rados.Error as e:
rados_client.shutdown()
self.skipTest("Failed to create pool: %s")
self.rados_client = rados_client
self.rbd_config = config
super(TestRBDStore, self).setUp()
**** CubicPower OpenStack Study ****
def tearDown(self):
self.rados_client.delete_pool(self.rbd_config['rbd_store_pool'])
self.rados_client.shutdown()
super(TestRBDStore, self).tearDown()
**** CubicPower OpenStack Study ****
def get_store(self, **kwargs):
store = glance.store.rbd.Store(context=kwargs.get('context'))
store.configure()
store.configure_add()
return store
**** CubicPower OpenStack Study ****
def stash_image(self, image_id, image_data):
fsid = self.rados_client.get_fsid()
pool = self.rbd_config['rbd_store_pool']
librbd = rbd.RBD()
# image_id must not be unicode since librbd doesn't handle it
image_id = str(image_id)
snap_name = 'snap'
with self.rados_client.open_ioctx(pool) as ioctx:
librbd.create(ioctx, image_id, len(image_data), old_format=False,
features=rbd.RBD_FEATURE_LAYERING)
with rbd.Image(ioctx, image_id) as image:
image.write(image_data, 0)
image.create_snap(snap_name)
return 'rbd://%s/%s/%s/%s' % (fsid, pool, image_id, snap_name)
**** CubicPower OpenStack Study ****
def test_unicode(self):
# librbd does not handle unicode, so make sure
# all paths through the rbd store convert a unicode image id
# and uri to ascii before passing it to librbd.
store = self.get_store()
image_id = unicode(str(uuid.uuid4()))
image_size = 300
image_data = six.StringIO('X' * image_size)
image_checksum = '41757066eaff7c4c6c965556b4d3c6c5'
uri, add_size, add_checksum = store.add(image_id,
image_data,
image_size)
uri = unicode(uri)
self.assertEqual(image_size, add_size)
self.assertEqual(image_checksum, add_checksum)
location = glance.store.location.Location(
self.store_name,
store.get_store_location_class(),
uri=uri,
image_id=image_id)
self.assertEqual(image_size, store.get_size(location))
get_iter, get_size = store.get(location)
self.assertEqual(image_size, get_size)
self.assertEqual('X' * image_size, ''.join(get_iter))
store.delete(location)
self.assertRaises(exception.NotFound, store.get, location)