¡@

Home 

OpenStack Study: test_nexenta.py

OpenStack Index

**** CubicPower OpenStack Study ****

#

# Copyright 2011 Nexenta Systems, 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.

"""

Unit tests for OpenStack Cinder volume driver

"""

import base64

import urllib2

import mox as mox_lib

from cinder import context

from cinder import db

from cinder import test

from cinder import units

from cinder.volume import configuration as conf

from cinder.volume.drivers import nexenta

from cinder.volume.drivers.nexenta import iscsi

from cinder.volume.drivers.nexenta import jsonrpc

from cinder.volume.drivers.nexenta import nfs

from cinder.volume.drivers.nexenta import utils

**** CubicPower OpenStack Study ****

class TestNexentaISCSIDriver(test.TestCase):

TEST_VOLUME_NAME = 'volume1'

TEST_VOLUME_NAME2 = 'volume2'

TEST_SNAPSHOT_NAME = 'snapshot1'

TEST_VOLUME_REF = {

'name': TEST_VOLUME_NAME,

'size': 1,

'id': '1',

'status': 'available'

}

TEST_VOLUME_REF2 = {

'name': TEST_VOLUME_NAME2,

'size': 1,

'id': '2',

'status': 'in-use'

}

TEST_SNAPSHOT_REF = {

'name': TEST_SNAPSHOT_NAME,

'volume_name': TEST_VOLUME_NAME,

}

**** CubicPower OpenStack Study ****

    def __init__(self, method):

        super(TestNexentaISCSIDriver, self).__init__(method)

**** CubicPower OpenStack Study ****

    def setUp(self):

        super(TestNexentaISCSIDriver, self).setUp()

        self.configuration = mox_lib.MockObject(conf.Configuration)

        self.configuration.nexenta_host = '1.1.1.1'

        self.configuration.nexenta_user = 'admin'

        self.configuration.nexenta_password = 'nexenta'

        self.configuration.nexenta_volume = 'cinder'

        self.configuration.nexenta_rest_port = 2000

        self.configuration.nexenta_rest_protocol = 'http'

        self.configuration.nexenta_iscsi_target_portal_port = 3260

        self.configuration.nexenta_target_prefix = 'iqn:'

        self.configuration.nexenta_target_group_prefix = 'cinder/'

        self.configuration.nexenta_blocksize = '8K'

        self.configuration.nexenta_sparse = True

        self.configuration.nexenta_rrmgr_compression = 1

        self.configuration.nexenta_rrmgr_tcp_buf_size = 1024

        self.configuration.nexenta_rrmgr_connections = 2

        self.nms_mock = self.mox.CreateMockAnything()

        for mod in ['volume', 'zvol', 'iscsitarget', 'appliance',

                    'stmf', 'scsidisk', 'snapshot']:

            setattr(self.nms_mock, mod, self.mox.CreateMockAnything())

        self.stubs.Set(jsonrpc, 'NexentaJSONProxy',

                       lambda *_, **__: self.nms_mock)

        self.drv = iscsi.NexentaISCSIDriver(configuration=self.configuration)

        self.drv.do_setup({})

**** CubicPower OpenStack Study ****

    def test_setup_error(self):

        self.nms_mock.volume.object_exists('cinder').AndReturn(True)

        self.mox.ReplayAll()

        self.drv.check_for_setup_error()

**** CubicPower OpenStack Study ****

    def test_setup_error_fail(self):

        self.nms_mock.volume.object_exists('cinder').AndReturn(False)

        self.mox.ReplayAll()

        self.assertRaises(LookupError, self.drv.check_for_setup_error)

**** CubicPower OpenStack Study ****

    def test_local_path(self):

        self.assertRaises(NotImplementedError, self.drv.local_path, '')

**** CubicPower OpenStack Study ****

    def test_create_volume(self):

        self.nms_mock.zvol.create('cinder/volume1', '1G', '8K', True)

        self.nms_mock.stmf.list_targets()

        self.nms_mock.iscsitarget.create_target({'target_name': 'iqn:volume1'})

        self.nms_mock.stmf.list_targetgroups()

        self.nms_mock.stmf.create_targetgroup('cinder/volume1')

        self.nms_mock.stmf.list_targetgroup_members('cinder/volume1')

        self.nms_mock.stmf.add_targetgroup_member('cinder/volume1',

                                                  'iqn:volume1')

        self.nms_mock.scsidisk.lu_exists('cinder/volume1')

        self.nms_mock.scsidisk.create_lu('cinder/volume1', {})

        self.nms_mock.scsidisk.lu_shared('cinder/volume1')

        self.nms_mock.scsidisk.add_lun_mapping_entry(

            'cinder/volume1', {'target_group': 'cinder/volume1', 'lun': '0'})

        self.mox.ReplayAll()

        self.drv.create_volume(self.TEST_VOLUME_REF)

**** CubicPower OpenStack Study ****

    def test_delete_volume(self):

        self.nms_mock.zvol.get_child_props('cinder/volume1',

                                           'origin').AndReturn({})

        self.nms_mock.zvol.destroy('cinder/volume1', '')

        self.mox.ReplayAll()

        self.drv.delete_volume(self.TEST_VOLUME_REF)

        self.mox.ResetAll()

        c = self.nms_mock.zvol.get_child_props('cinder/volume1', 'origin')

        c.AndReturn({'origin': 'cinder/volume0@snapshot'})

        self.nms_mock.zvol.destroy('cinder/volume1', '')

        self.mox.ReplayAll()

        self.drv.delete_volume(self.TEST_VOLUME_REF)

        self.mox.ResetAll()

        c = self.nms_mock.zvol.get_child_props('cinder/volume1', 'origin')

        c.AndReturn({'origin': 'cinder/volume0@cinder-clone-snapshot-1'})

        self.nms_mock.zvol.destroy('cinder/volume1', '')

        self.nms_mock.snapshot.destroy(

            'cinder/volume0@cinder-clone-snapshot-1', '')

        self.mox.ReplayAll()

        self.drv.delete_volume(self.TEST_VOLUME_REF)

        self.mox.ResetAll()

**** CubicPower OpenStack Study ****

    def test_create_cloned_volume(self):

        vol = self.TEST_VOLUME_REF2

        src_vref = self.TEST_VOLUME_REF

        snapshot = {

            'volume_name': src_vref['name'],

            'name': 'cinder-clone-snapshot-%s' % vol['id'],

        }

        self.nms_mock.zvol.create_snapshot('cinder/%s' % src_vref['name'],

                                           snapshot['name'], '')

        self.nms_mock.zvol.clone('cinder/%s@%s' % (src_vref['name'],

                                                   snapshot['name']),

                                 'cinder/%s' % vol['name'])

        self.mox.ReplayAll()

        self.drv.create_cloned_volume(vol, src_vref)

**** CubicPower OpenStack Study ****

    def test_migrate_volume(self):

        volume = self.TEST_VOLUME_REF

        host = {

            'capabilities': {

                'vendor_name': 'Nexenta',

                'location_info': 'NexentaISCSIDriver:1.1.1.1:cinder',

                'free_capacity_gb': 1,

                'iscsi_target_portal_port': 3260,

                'nms_url': 'http://admin:password@1.1.1.1:2000'

            }

        }

        snapshot = {

            'volume_name': volume['name'],

            'name': 'cinder-migrate-snapshot-%s' % volume['id'],

        }

        self.nms_mock.appliance.ssh_list_bindings().AndReturn([])

        self.nms_mock.zvol.create_snapshot('cinder/%s' % volume['name'],

                                           snapshot['name'], '')

        src = '%(volume)s/%(zvol)s@%(snapshot)s' % {

            'volume': 'cinder',

            'zvol': volume['name'],

            'snapshot': snapshot['name']

        }

        dst = '1.1.1.1:cinder'

        cmd = ' '.join(['rrmgr -s zfs -c 1 -q -e -w 1024 -n 2', src, dst])

        self.nms_mock.appliance.execute(cmd)

        snapshot_name = 'cinder/%(volume)s@%(snapshot)s' % {

            'volume': volume['name'],

            'snapshot': snapshot['name']

        }

        self.nms_mock.snapshot.destroy(snapshot_name, '')

        volume_name = 'cinder/%s' % volume['name']

        self.nms_mock.zvol.get_child_props(volume_name,

                                           'origin').AndReturn(None)

        self.nms_mock.zvol.destroy(volume_name, '')

        self.nms_mock.snapshot.destroy('cinder/%(volume)s@%(snapshot)s' % {

            'volume': volume['name'],

            'snapshot': snapshot['name']

        }, '')

        self.mox.ReplayAll()

        self.drv.migrate_volume(None, volume, host)

**** CubicPower OpenStack Study ****

    def test_create_snapshot(self):

        self.nms_mock.zvol.create_snapshot('cinder/volume1', 'snapshot1', '')

        self.mox.ReplayAll()

        self.drv.create_snapshot(self.TEST_SNAPSHOT_REF)

**** CubicPower OpenStack Study ****

    def test_create_volume_from_snapshot(self):

        self.nms_mock.zvol.clone('cinder/volume1@snapshot1', 'cinder/volume2')

        self.mox.ReplayAll()

        self.drv.create_volume_from_snapshot(self.TEST_VOLUME_REF2,

                                             self.TEST_SNAPSHOT_REF)

**** CubicPower OpenStack Study ****

    def test_delete_snapshot(self):

        self.nms_mock.snapshot.destroy('cinder/volume1@snapshot1', '')

        self.mox.ReplayAll()

        self.drv.delete_snapshot(self.TEST_SNAPSHOT_REF)

        self.mox.ResetAll()

        # Check that exception not raised if snapshot does not exist

        mock = self.nms_mock.snapshot.destroy('cinder/volume1@snapshot1', '')

        mock.AndRaise(nexenta.NexentaException(

            'Snapshot cinder/volume1@snapshot1 does not exist'))

        self.mox.ReplayAll()

        self.drv.delete_snapshot(self.TEST_SNAPSHOT_REF)

    _CREATE_EXPORT_METHODS = [

        ('stmf', 'list_targets', tuple(), [], False, ),

        ('iscsitarget', 'create_target', ({'target_name': 'iqn:volume1'},),

            u'Unable to create iscsi target\n'

            u' iSCSI target iqn.1986-03.com.sun:02:cinder-volume1 already'

            u' configured\n'

            u' itadm create-target failed with error 17\n', True, ),

        ('stmf', 'list_targetgroups', tuple(), [], False, ),

        ('stmf', 'create_targetgroup', ('cinder/volume1',),

            u'Unable to create targetgroup: stmfadm: cinder/volume1:'

            u' already exists\n', True, ),

        ('stmf', 'list_targetgroup_members', ('cinder/volume1', ), [],

         False, ),

        ('stmf', 'add_targetgroup_member', ('cinder/volume1', 'iqn:volume1'),

            u'Unable to add member to targetgroup: stmfadm:'

            u' iqn.1986-03.com.sun:02:cinder-volume1: already exists\n',

            True, ),

        ('scsidisk', 'lu_exists', ('cinder/volume1', ), 0, False, ),

        ('scsidisk', 'create_lu', ('cinder/volume1', {}),

            u"Unable to create lu with zvol 'cinder/volume1':\n"

            u" sbdadm: filename /dev/zvol/rdsk/cinder/volume1: in use\n",

            True, ),

        ('scsidisk', 'lu_shared', ('cinder/volume1', ), 0, False, ),

        ('scsidisk', 'add_lun_mapping_entry', ('cinder/volume1', {

            'target_group': 'cinder/volume1', 'lun': '0'}),

            u"Unable to add view to zvol 'cinder/volume1' (LUNs in use: ):\n"

            u" stmfadm: view entry exists\n", True, ),

    ]

**** CubicPower OpenStack Study ****

    def _stub_export_method(self, module, method, args, error, raise_exception,

                            fail=False):

        m = getattr(self.nms_mock, module)

        m = getattr(m, method)

        mock = m(*args)

        if raise_exception and fail:

            mock.AndRaise(nexenta.NexentaException(error))

        else:

            mock.AndReturn(error)

**** CubicPower OpenStack Study ****

    def _stub_all_export_methods(self, fail=False):

        for params in self._CREATE_EXPORT_METHODS:

            self._stub_export_method(*params, fail=fail)

**** CubicPower OpenStack Study ****

    def test_create_export(self):

        self._stub_all_export_methods()

        self.mox.ReplayAll()

        retval = self.drv.create_export({}, self.TEST_VOLUME_REF)

        location = '%(host)s:%(port)s,1 %(prefix)s%(volume)s 0' % {

            'host': self.configuration.nexenta_host,

            'port': self.configuration.nexenta_iscsi_target_portal_port,

            'prefix': self.configuration.nexenta_target_prefix,

            'volume': self.TEST_VOLUME_NAME

        }

        self.assertEqual(retval, {'provider_location': location})

**** CubicPower OpenStack Study ****

    def __get_test(i):

        def _test_create_export_fail(self):

            for params in self._CREATE_EXPORT_METHODS[:i]:

                self._stub_export_method(*params)

            self._stub_export_method(*self._CREATE_EXPORT_METHODS[i],

                                     fail=True)

            self.mox.ReplayAll()

            self.assertRaises(nexenta.NexentaException,

                              self.drv.create_export,

                              {},

                              self.TEST_VOLUME_REF)

        return _test_create_export_fail

    for i in range(len(_CREATE_EXPORT_METHODS)):

        if i % 2:

            locals()['test_create_export_fail_%d' % i] = __get_test(i)

**** CubicPower OpenStack Study ****

        def _test_create_export_fail(self):

            for params in self._CREATE_EXPORT_METHODS[:i]:

                self._stub_export_method(*params)

            self._stub_export_method(*self._CREATE_EXPORT_METHODS[i],

                                     fail=True)

            self.mox.ReplayAll()

            self.assertRaises(nexenta.NexentaException,

                              self.drv.create_export,

                              {},

                              self.TEST_VOLUME_REF)

        return _test_create_export_fail

    for i in range(len(_CREATE_EXPORT_METHODS)):

        if i % 2:

            locals()['test_create_export_fail_%d' % i] = __get_test(i)

**** CubicPower OpenStack Study ****

    def test_ensure_export(self):

        self._stub_all_export_methods(fail=True)

        self.mox.ReplayAll()

        self.drv.ensure_export({}, self.TEST_VOLUME_REF)

**** CubicPower OpenStack Study ****

    def test_remove_export(self):

        self.nms_mock.scsidisk.delete_lu('cinder/volume1')

        self.nms_mock.stmf.destroy_targetgroup('cinder/volume1')

        self.nms_mock.iscsitarget.delete_target('iqn:volume1')

        self.mox.ReplayAll()

        self.drv.remove_export({}, self.TEST_VOLUME_REF)

**** CubicPower OpenStack Study ****

    def test_remove_export_fail_0(self):

        self.nms_mock.scsidisk.delete_lu('cinder/volume1')

        self.nms_mock.stmf.destroy_targetgroup(

            'cinder/volume1').AndRaise(nexenta.NexentaException())

        self.nms_mock.iscsitarget.delete_target('iqn:volume1')

        self.mox.ReplayAll()

        self.drv.remove_export({}, self.TEST_VOLUME_REF)

**** CubicPower OpenStack Study ****

    def test_remove_export_fail_1(self):

        self.nms_mock.scsidisk.delete_lu('cinder/volume1')

        self.nms_mock.stmf.destroy_targetgroup('cinder/volume1')

        self.nms_mock.iscsitarget.delete_target(

            'iqn:volume1').AndRaise(nexenta.NexentaException())

        self.mox.ReplayAll()

        self.drv.remove_export({}, self.TEST_VOLUME_REF)

**** CubicPower OpenStack Study ****

    def test_get_volume_stats(self):

        stats = {'size': '5368709120G',

                 'used': '5368709120G',

                 'available': '5368709120G',

                 'health': 'ONLINE'}

        self.nms_mock.volume.get_child_props(

            self.configuration.nexenta_volume,

            'health|size|used|available').AndReturn(stats)

        self.mox.ReplayAll()

        stats = self.drv.get_volume_stats(True)

        self.assertEqual(stats['storage_protocol'], 'iSCSI')

        self.assertEqual(stats['total_capacity_gb'], 5368709120.0)

        self.assertEqual(stats['free_capacity_gb'], 5368709120.0)

        self.assertEqual(stats['reserved_percentage'], 0)

        self.assertEqual(stats['QoS_support'], False)

**** CubicPower OpenStack Study ****

class TestNexentaJSONRPC(test.TestCase):

HOST = 'example.com'

URL = 'http://%s/' % HOST

URL_S = 'https://%s/' % HOST

USER = 'user'

PASSWORD = 'password'

HEADERS = {

'Authorization':

'Basic %s' % base64.b64encode('%s:%s' % (USER, PASSWORD)),

'Content-Type': 'application/json'

}

REQUEST = 'the request'

**** CubicPower OpenStack Study ****

    def setUp(self):

        super(TestNexentaJSONRPC, self).setUp()

        self.proxy = jsonrpc.NexentaJSONProxy(

            'http', self.HOST, 2000, '/', self.USER, self.PASSWORD, auto=True)

        self.mox.StubOutWithMock(urllib2, 'Request', True)

        self.mox.StubOutWithMock(urllib2, 'urlopen')

        self.resp_mock = self.mox.CreateMockAnything()

        self.resp_info_mock = self.mox.CreateMockAnything()

        self.resp_mock.info().AndReturn(self.resp_info_mock)

        urllib2.urlopen(self.REQUEST).AndReturn(self.resp_mock)

**** CubicPower OpenStack Study ****

    def test_call(self):

        urllib2.Request(

            'http://%s:2000/' % self.HOST,

            '{"object": null, "params": ["arg1", "arg2"], "method": null}',

            self.HEADERS).AndReturn(self.REQUEST)

        self.resp_info_mock.status = ''

        self.resp_mock.read().AndReturn(

            '{"error": null, "result": "the result"}')

        self.mox.ReplayAll()

        result = self.proxy('arg1', 'arg2')

        self.assertEqual("the result", result)

**** CubicPower OpenStack Study ****

    def test_call_deep(self):

        urllib2.Request(

            'http://%s:2000/' % self.HOST,

            '{"object": "obj1.subobj", "params": ["arg1", "arg2"],'

            ' "method": "meth"}',

            self.HEADERS).AndReturn(self.REQUEST)

        self.resp_info_mock.status = ''

        self.resp_mock.read().AndReturn(

            '{"error": null, "result": "the result"}')

        self.mox.ReplayAll()

        result = self.proxy.obj1.subobj.meth('arg1', 'arg2')

        self.assertEqual("the result", result)

**** CubicPower OpenStack Study ****

    def test_call_auto(self):

        urllib2.Request(

            'http://%s:2000/' % self.HOST,

            '{"object": null, "params": ["arg1", "arg2"], "method": null}',

            self.HEADERS).AndReturn(self.REQUEST)

        urllib2.Request(

            'https://%s:2000/' % self.HOST,

            '{"object": null, "params": ["arg1", "arg2"], "method": null}',

            self.HEADERS).AndReturn(self.REQUEST)

        self.resp_info_mock.status = 'EOF in headers'

        self.resp_mock.read().AndReturn(

            '{"error": null, "result": "the result"}')

        urllib2.urlopen(self.REQUEST).AndReturn(self.resp_mock)

        self.mox.ReplayAll()

        result = self.proxy('arg1', 'arg2')

        self.assertEqual("the result", result)

**** CubicPower OpenStack Study ****

    def test_call_error(self):

        urllib2.Request(

            'http://%s:2000/' % self.HOST,

            '{"object": null, "params": ["arg1", "arg2"], "method": null}',

            self.HEADERS).AndReturn(self.REQUEST)

        self.resp_info_mock.status = ''

        self.resp_mock.read().AndReturn(

            '{"error": {"message": "the error"}, "result": "the result"}')

        self.mox.ReplayAll()

        self.assertRaises(jsonrpc.NexentaJSONException,

                          self.proxy, 'arg1', 'arg2')

**** CubicPower OpenStack Study ****

    def test_call_fail(self):

        urllib2.Request(

            'http://%s:2000/' % self.HOST,

            '{"object": null, "params": ["arg1", "arg2"], "method": null}',

            self.HEADERS).AndReturn(self.REQUEST)

        self.resp_info_mock.status = 'EOF in headers'

        self.proxy.auto = False

        self.mox.ReplayAll()

        self.assertRaises(jsonrpc.NexentaJSONException,

                          self.proxy, 'arg1', 'arg2')

**** CubicPower OpenStack Study ****

class TestNexentaNfsDriver(test.TestCase):

TEST_EXPORT1 = 'host1:/volumes/stack/share'

TEST_NMS1 = 'http://admin:nexenta@host1:2000'

TEST_EXPORT2 = 'host2:/volumes/stack/share'

TEST_NMS2 = 'http://admin:nexenta@host2:2000'

TEST_EXPORT2_OPTIONS = '-o intr'

TEST_FILE_NAME = 'test.txt'

TEST_SHARES_CONFIG_FILE = '/etc/cinder/nexenta-shares.conf'

TEST_SHARE_SVC = 'svc:/network/nfs/server:

**** CubicPower OpenStack Study ****

    def _create_volume_db_entry(self):

        vol = {

            'id': '1',

            'size': 1,

            'status': 'available',

            'provider_location': self.TEST_EXPORT1

        }

        return db.volume_create(self.ctxt, vol)['id']

**** CubicPower OpenStack Study ****

    def setUp(self):

        super(TestNexentaNfsDriver, self).setUp()

        self.ctxt = context.get_admin_context()

        self.configuration = mox_lib.MockObject(conf.Configuration)

        self.configuration.nexenta_shares_config = None

        self.configuration.nexenta_mount_point_base = '$state_path/mnt'

        self.configuration.nexenta_sparsed_volumes = True

        self.configuration.nexenta_volume_compression = 'on'

        self.configuration.nfs_mount_point_base = '/mnt/test'

        self.configuration.nfs_mount_options = None

        self.configuration.nexenta_nms_cache_volroot = False

        self.nms_mock = self.mox.CreateMockAnything()

        for mod in ('appliance', 'folder', 'server', 'volume', 'netstorsvc',

                    'snapshot'):

            setattr(self.nms_mock, mod, self.mox.CreateMockAnything())

        self.nms_mock.__hash__ = lambda *_, **__: 1

        self.stubs.Set(jsonrpc, 'NexentaJSONProxy',

                       lambda *_, **__: self.nms_mock)

        self.drv = nfs.NexentaNfsDriver(configuration=self.configuration)

        self.drv.shares = {}

        self.drv.share2nms = {}

**** CubicPower OpenStack Study ****

    def test_check_for_setup_error(self):

        self.drv.share2nms = {

            'host1:/volumes/stack/share': self.nms_mock

        }

        self.nms_mock.server.get_prop('volroot').AndReturn('/volumes')

        self.nms_mock.volume.object_exists('stack').AndReturn(True)

        self.nms_mock.folder.object_exists('stack/share').AndReturn(True)

        share_opts = {

            'read_write': '*',

            'read_only': '',

            'root': 'nobody',

            'extra_options': 'anon=0',

            'recursive': 'true',

            'anonymous_rw': 'true',

        }

        self.nms_mock.netstorsvc.share_folder(

            'svc:/network/nfs/server:default', 'stack/share', share_opts)

        self.mox.ReplayAll()

        self.drv.check_for_setup_error()

        self.mox.ResetAll()

        self.nms_mock.server.get_prop('volroot').AndReturn('/volumes')

        self.nms_mock.volume.object_exists('stack').AndReturn(False)

        self.mox.ReplayAll()

        self.assertRaises(LookupError, self.drv.check_for_setup_error)

        self.mox.ResetAll()

        self.nms_mock.server.get_prop('volroot').AndReturn('/volumes')

        self.nms_mock.volume.object_exists('stack').AndReturn(True)

        self.nms_mock.folder.object_exists('stack/share').AndReturn(False)

        self.mox.ReplayAll()

        self.assertRaises(LookupError, self.drv.check_for_setup_error)

**** CubicPower OpenStack Study ****

    def test_initialize_connection(self):

        self.drv.shares = {

            self.TEST_EXPORT1: None

        }

        volume = {

            'provider_location': self.TEST_EXPORT1,

            'name': 'volume'

        }

        result = self.drv.initialize_connection(volume, None)

        self.assertEqual(result['data']['export'],

                         '%s/volume' % self.TEST_EXPORT1)

**** CubicPower OpenStack Study ****

    def test_do_create_volume(self):

        volume = {

            'provider_location': self.TEST_EXPORT1,

            'size': 1,

            'name': 'volume-1'

        }

        self.drv.shares = {self.TEST_EXPORT1: None}

        self.drv.share2nms = {self.TEST_EXPORT1: self.nms_mock}

        compression = self.configuration.nexenta_volume_compression

        self.nms_mock.server.get_prop('volroot').AndReturn('/volumes')

        self.nms_mock.folder.create_with_props(

            'stack', 'share/volume-1', {'compression': compression})

        self.nms_mock.netstorsvc.share_folder(self.TEST_SHARE_SVC,

                                              'stack/share/volume-1',

                                              self.TEST_SHARE_OPTS)

        self.nms_mock.appliance.execute(

            'dd if=/dev/zero of=/volumes/stack/share/volume-1/volume bs=1M '

            'count=0 seek=1024'

        )

        self.nms_mock.appliance.execute('chmod ugo+rw '

                                        '/volumes/stack/share/volume-1/volume')

        self.mox.ReplayAll()

        self.drv._do_create_volume(volume)

        self.mox.ResetAll()

        self.nms_mock.server.get_prop('volroot').AndReturn('/volumes')

        self.nms_mock.folder.create_with_props(

            'stack', 'share/volume-1', {'compression': compression})

        self.nms_mock.netstorsvc.share_folder(

            self.TEST_SHARE_SVC, 'stack/share/volume-1',

            self.TEST_SHARE_OPTS).AndRaise(nexenta.NexentaException('-'))

        self.nms_mock.folder.destroy('stack/share/volume-1')

        self.mox.ReplayAll()

        self.assertRaises(nexenta.NexentaException, self.drv._do_create_volume,

                          volume)

**** CubicPower OpenStack Study ****

    def test_create_sparsed_file(self):

        self.nms_mock.appliance.execute('dd if=/dev/zero of=/tmp/path bs=1M '

                                        'count=0 seek=1024')

        self.mox.ReplayAll()

        self.drv._create_sparsed_file(self.nms_mock, '/tmp/path', 1)

**** CubicPower OpenStack Study ****

    def test_create_regular_file(self):

        self.nms_mock.appliance.execute('dd if=/dev/zero of=/tmp/path bs=1M '

                                        'count=1024')

        self.mox.ReplayAll()

        self.drv._create_regular_file(self.nms_mock, '/tmp/path', 1)

**** CubicPower OpenStack Study ****

    def test_set_rw_permissions_for_all(self):

        path = '/tmp/path'

        self.nms_mock.appliance.execute('chmod ugo+rw %s' % path)

        self.mox.ReplayAll()

        self.drv._set_rw_permissions_for_all(self.nms_mock, path)

**** CubicPower OpenStack Study ****

    def test_local_path(self):

        volume = {'provider_location': self.TEST_EXPORT1, 'name': 'volume-1'}

        path = self.drv.local_path(volume)

        self.assertEqual(

            path,

            '$state_path/mnt/b3f660847a52b29ac330d8555e4ad669/volume-1/volume'

        )

**** CubicPower OpenStack Study ****

    def test_remote_path(self):

        volume = {'provider_location': self.TEST_EXPORT1, 'name': 'volume-1'}

        path = self.drv.remote_path(volume)

        self.assertEqual(path, '/volumes/stack/share/volume-1/volume')

**** CubicPower OpenStack Study ****

    def test_share_folder(self):

        path = 'stack/share/folder'

        self.nms_mock.netstorsvc.share_folder(self.TEST_SHARE_SVC, path,

                                              self.TEST_SHARE_OPTS)

        self.mox.ReplayAll()

        self.drv._share_folder(self.nms_mock, 'stack', 'share/folder')

**** CubicPower OpenStack Study ****

    def test_load_shares_config(self):

        self.drv.configuration.nfs_shares_config = self.TEST_SHARES_CONFIG_FILE

        self.mox.StubOutWithMock(self.drv, '_read_config_file')

        config_data = [

            '%s %s' % (self.TEST_EXPORT1, self.TEST_NMS1),

            '# %s %s' % (self.TEST_EXPORT2, self.TEST_NMS2),

            '',

            '%s %s %s' % (self.TEST_EXPORT2, self.TEST_NMS2,

                          self.TEST_EXPORT2_OPTIONS)

        ]

        self.drv._read_config_file(self.TEST_SHARES_CONFIG_FILE).\

            AndReturn(config_data)

        self.mox.ReplayAll()

        self.drv._load_shares_config(self.drv.configuration.nfs_shares_config)

        self.assertIn(self.TEST_EXPORT1, self.drv.shares)

        self.assertIn(self.TEST_EXPORT2, self.drv.shares)

        self.assertEqual(len(self.drv.shares), 2)

        self.assertIn(self.TEST_EXPORT1, self.drv.share2nms)

        self.assertIn(self.TEST_EXPORT2, self.drv.share2nms)

        self.assertEqual(len(self.drv.share2nms.keys()), 2)

        self.assertEqual(self.drv.shares[self.TEST_EXPORT2],

                         self.TEST_EXPORT2_OPTIONS)

        self.mox.VerifyAll()

**** CubicPower OpenStack Study ****

    def test_get_capacity_info(self):

        self.drv.share2nms = {self.TEST_EXPORT1: self.nms_mock}

        self.nms_mock.server.get_prop('volroot').AndReturn('/volumes')

        self.nms_mock.folder.get_child_props('stack/share', '').AndReturn({

            'available': '1G',

            'used': '2G'

        })

        self.mox.ReplayAll()

        total, free, allocated = self.drv._get_capacity_info(self.TEST_EXPORT1)

        self.assertEqual(total, 3 * units.GiB)

        self.assertEqual(free, units.GiB)

        self.assertEqual(allocated, 2 * units.GiB)

**** CubicPower OpenStack Study ****

    def test_get_share_datasets(self):

        self.drv.share2nms = {self.TEST_EXPORT1: self.nms_mock}

        self.nms_mock.server.get_prop('volroot').AndReturn('/volumes')

        self.mox.ReplayAll()

        volume_name, folder_name = \

            self.drv._get_share_datasets(self.TEST_EXPORT1)

        self.assertEqual(volume_name, 'stack')

        self.assertEqual(folder_name, 'share')

**** CubicPower OpenStack Study ****

    def test_delete_snapshot(self):

        self.drv.share2nms = {self.TEST_EXPORT1: self.nms_mock}

        self._create_volume_db_entry()

        self.nms_mock.server.get_prop('volroot').AndReturn('/volumes')

        self.nms_mock.snapshot.destroy('stack/share/volume-1@snapshot1', '')

        self.mox.ReplayAll()

        self.drv.delete_snapshot({'volume_id': '1', 'name': 'snapshot1'})

        self.mox.ResetAll()

        # Check that exception not raised if snapshot does not exist on

        # NexentaStor appliance.

        self.nms_mock.server.get_prop('volroot').AndReturn('/volumes')

        mock = self.nms_mock.snapshot.destroy('stack/share/volume-1@snapshot1',

                                              '')

        mock.AndRaise(nexenta.NexentaException("Snapshot does not exist"))

        self.mox.ReplayAll()

        self.drv.delete_snapshot({'volume_id': '1', 'name': 'snapshot1'})

        self.mox.ResetAll()

**** CubicPower OpenStack Study ****

    def test_delete_volume(self):

        self.drv.share2nms = {self.TEST_EXPORT1: self.nms_mock}

        self._create_volume_db_entry()

        self.drv._ensure_share_mounted = lambda *_, **__: 0

        self.drv._execute = lambda *_, **__: 0

        self.nms_mock.server.get_prop('volroot').AndReturn('/volumes')

        self.nms_mock.folder.get_child_props('stack/share/volume-1',

                                             'origin').AndReturn(None)

        self.nms_mock.folder.destroy('stack/share/volume-1', '-r')

        self.mox.ReplayAll()

        self.drv.delete_volume({

            'id': '1',

            'name': 'volume-1',

            'provider_location': self.TEST_EXPORT1

        })

        self.mox.ResetAll()

        # Check that exception not raised if folder does not exist on

        # NexentaStor appliance.

        self.nms_mock.server.get_prop('volroot').AndReturn('/volumes')

        self.nms_mock.folder.get_child_props('stack/share/volume-1',

                                             'origin').AndReturn(None)

        mock = self.nms_mock.folder.destroy('stack/share/volume-1', '-r')

        mock.AndRaise(nexenta.NexentaException("Folder does not exist"))

        self.mox.ReplayAll()

        self.drv.delete_volume({

            'id': '1',

            'name': 'volume-1',

            'provider_location': self.TEST_EXPORT1

        })

        self.mox.ResetAll()

**** CubicPower OpenStack Study ****

class TestNexentaUtils(test.TestCase):

**** CubicPower OpenStack Study ****

    def test_str2size(self):

        values_to_test = (

            # Test empty value

            (None, 0),

            ('', 0),

            ('0', 0),

            ('12', 12),

            # Test int and long values

            (10, 10),

            (long(10), 10),

            # Test bytes string

            ('1b', 1),

            ('1B', 1),

            ('1023b', 1023),

            ('0B', 0),

            # Test other units

            ('1M', units.MiB),

            ('1.0M', units.MiB),

        )

        for value, result in values_to_test:

            self.assertEqual(utils.str2size(value), result)

        # Invalid format value

        self.assertRaises(ValueError, utils.str2size, 'A')

**** CubicPower OpenStack Study ****

    def test_str2gib_size(self):

        self.assertEqual(utils.str2gib_size('1024M'), 1)

        self.assertEqual(utils.str2gib_size('300M'),

                         300 * units.MiB // units.GiB)

        self.assertEqual(utils.str2gib_size('1.2T'),

                         1.2 * units.TiB // units.GiB)

        self.assertRaises(ValueError, utils.str2gib_size, 'A')

**** CubicPower OpenStack Study ****

    def test_parse_nms_url(self):

        urls = (

            ('auto://192.168.1.1/', (True, 'http', 'admin', 'nexenta',

                                     '192.168.1.1', '2000', '/rest/nms/')),

            ('http://192.168.1.1/', (False, 'http', 'admin', 'nexenta',

                                     '192.168.1.1', '2000', '/rest/nms/')),

            ('http://192.168.1.1:8080', (False, 'http', 'admin', 'nexenta',

                                         '192.168.1.1', '8080', '/rest/nms/')),

            ('https://root:password@192.168.1.1:8080',

             (False, 'https', 'root', 'password', '192.168.1.1', '8080',

              '/rest/nms/')),

        )

        for url, result in urls:

            self.assertEqual(utils.parse_nms_url(url), result)