Modify the configuration mode of FusionStorage Cinder Driver
Modify the configuration mode by using existing standard config options. Change-Id: I033a739c08c01867660c00ea44642e78ddf3e3ea
This commit is contained in:
parent
a8d995c2f2
commit
c9aa071b32
@ -15,6 +15,7 @@
|
|||||||
import ddt
|
import ddt
|
||||||
|
|
||||||
import mock
|
import mock
|
||||||
|
import os
|
||||||
import shutil
|
import shutil
|
||||||
import tempfile
|
import tempfile
|
||||||
|
|
||||||
@ -43,52 +44,108 @@ class FusionStorageConfTestCase(test.TestCase):
|
|||||||
self.tmp_dir + '/cinder.conf')
|
self.tmp_dir + '/cinder.conf')
|
||||||
|
|
||||||
config = configparser.ConfigParser()
|
config = configparser.ConfigParser()
|
||||||
config.add_section('storage')
|
config.add_section('backend_name')
|
||||||
config.set('storage', 'RestURL', 'https://fake_rest_site')
|
config.set('backend_name', 'dsware_rest_url', 'https://fake_rest_site')
|
||||||
config.set('storage', 'UserName', 'fake_user')
|
config.set('backend_name', 'san_login', 'fake_user')
|
||||||
config.set('storage', 'Password', 'fake_passwd')
|
config.set('backend_name', 'san_password', 'fake_passwd')
|
||||||
config.set('storage', 'StoragePool', 'fake_pool')
|
config.set('backend_name', 'dsware_storage_pools', 'fake_pool')
|
||||||
|
|
||||||
config.add_section('manager_ip')
|
config.add_section('manager_ip')
|
||||||
config.set('manager_ip', 'fake_host', 'fake_ip')
|
config.set('manager_ip', 'fake_host', 'fake_ip')
|
||||||
config.write(open(self.conf.cinder_fusionstorage_conf_file, 'w'))
|
config.write(open(self.conf.cinder_fusionstorage_conf_file, 'w'))
|
||||||
|
|
||||||
def test_update_config_value(self):
|
@mock.patch.object(fs_conf.FusionStorageConf, '_encode_authentication')
|
||||||
|
@mock.patch.object(fs_conf.FusionStorageConf, '_pools_name')
|
||||||
|
@mock.patch.object(fs_conf.FusionStorageConf, '_san_address')
|
||||||
|
@mock.patch.object(fs_conf.FusionStorageConf, '_san_user')
|
||||||
|
@mock.patch.object(fs_conf.FusionStorageConf, '_san_password')
|
||||||
|
def test_update_config_value(self, mock_san_password, mock_san_user,
|
||||||
|
mock_san_address, mock_pools_name,
|
||||||
|
mock_encode_authentication):
|
||||||
|
self.fusionstorage_conf.update_config_value()
|
||||||
|
mock_encode_authentication.assert_called_once_with()
|
||||||
|
mock_pools_name.assert_called_once_with()
|
||||||
|
mock_san_address.assert_called_once_with()
|
||||||
|
mock_san_user.assert_called_once_with()
|
||||||
|
mock_san_password.assert_called_once_with()
|
||||||
|
|
||||||
|
@mock.patch.object(os.path, 'exists')
|
||||||
|
def test__encode_authentication(self, mock_exists):
|
||||||
config = configparser.ConfigParser()
|
config = configparser.ConfigParser()
|
||||||
config.read(self.conf.cinder_fusionstorage_conf_file)
|
config.read(self.conf.cinder_fusionstorage_conf_file)
|
||||||
storage_info = {'RestURL': config.get('storage', 'RestURL'),
|
mock_exists.return_value = False
|
||||||
'UserName': config.get('storage', 'UserName'),
|
|
||||||
'Password': config.get('storage', 'Password'),
|
|
||||||
'StoragePool': config.get('storage', 'StoragePool')}
|
|
||||||
|
|
||||||
|
user_name = 'fake_user'
|
||||||
self.mock_object(
|
self.mock_object(
|
||||||
self.fusionstorage_conf.configuration, 'safe_get',
|
self.fusionstorage_conf.configuration, 'safe_get',
|
||||||
return_value=storage_info)
|
return_value=user_name)
|
||||||
|
self.fusionstorage_conf._encode_authentication()
|
||||||
|
|
||||||
self.fusionstorage_conf.update_config_value()
|
password = 'fake_passwd'
|
||||||
|
self.mock_object(
|
||||||
|
self.fusionstorage_conf.configuration, 'safe_get',
|
||||||
|
return_value=password)
|
||||||
|
self.fusionstorage_conf._encode_authentication()
|
||||||
|
|
||||||
|
@mock.patch.object(os.path, 'exists')
|
||||||
|
@mock.patch.object(configparser.ConfigParser, 'set')
|
||||||
|
def test__rewrite_conf(self, mock_set, mock_exists):
|
||||||
|
mock_exists.return_value = False
|
||||||
|
mock_set.return_value = "success"
|
||||||
|
self.fusionstorage_conf._rewrite_conf('fake_name', 'fake_pwd')
|
||||||
|
|
||||||
|
def test__san_address(self):
|
||||||
|
address = 'https://fake_rest_site'
|
||||||
|
self.mock_object(
|
||||||
|
self.fusionstorage_conf.configuration, 'safe_get',
|
||||||
|
return_value=address)
|
||||||
|
self.fusionstorage_conf._san_address()
|
||||||
self.assertEqual('https://fake_rest_site',
|
self.assertEqual('https://fake_rest_site',
|
||||||
self.fusionstorage_conf.configuration.san_address)
|
self.fusionstorage_conf.configuration.san_address)
|
||||||
|
|
||||||
|
def test__san_user(self):
|
||||||
|
user = '!&&&ZmFrZV91c2Vy'
|
||||||
|
self.mock_object(
|
||||||
|
self.fusionstorage_conf.configuration, 'safe_get',
|
||||||
|
return_value=user)
|
||||||
|
self.fusionstorage_conf._san_user()
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
'fake_user', self.fusionstorage_conf.configuration.san_user)
|
'fake_user', self.fusionstorage_conf.configuration.san_user)
|
||||||
|
|
||||||
|
user = 'fake_user_2'
|
||||||
|
self.mock_object(
|
||||||
|
self.fusionstorage_conf.configuration, 'safe_get',
|
||||||
|
return_value=user)
|
||||||
|
self.fusionstorage_conf._san_user()
|
||||||
|
self.assertEqual(
|
||||||
|
'fake_user_2', self.fusionstorage_conf.configuration.san_user)
|
||||||
|
|
||||||
|
def test__san_password(self):
|
||||||
|
password = '!&&&ZmFrZV9wYXNzd2Q='
|
||||||
|
self.mock_object(
|
||||||
|
self.fusionstorage_conf.configuration, 'safe_get',
|
||||||
|
return_value=password)
|
||||||
|
self.fusionstorage_conf._san_password()
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
'fake_passwd', self.fusionstorage_conf.configuration.san_password)
|
'fake_passwd', self.fusionstorage_conf.configuration.san_password)
|
||||||
|
|
||||||
|
password = 'fake_passwd_2'
|
||||||
|
self.mock_object(
|
||||||
|
self.fusionstorage_conf.configuration, 'safe_get',
|
||||||
|
return_value=password)
|
||||||
|
self.fusionstorage_conf._san_password()
|
||||||
|
self.assertEqual('fake_passwd_2',
|
||||||
|
self.fusionstorage_conf.configuration.san_password)
|
||||||
|
|
||||||
|
def test__pools_name(self):
|
||||||
|
pools_name = 'fake_pool'
|
||||||
|
self.mock_object(
|
||||||
|
self.fusionstorage_conf.configuration, 'safe_get',
|
||||||
|
return_value=pools_name)
|
||||||
|
self.fusionstorage_conf._pools_name()
|
||||||
self.assertListEqual(
|
self.assertListEqual(
|
||||||
['fake_pool'], self.fusionstorage_conf.configuration.pools_name)
|
['fake_pool'], self.fusionstorage_conf.configuration.pools_name)
|
||||||
|
|
||||||
def test__encode_authentication(self):
|
|
||||||
config = configparser.ConfigParser()
|
|
||||||
config.read(self.conf.cinder_fusionstorage_conf_file)
|
|
||||||
|
|
||||||
storage_info = {'RestURL': config.get('storage', 'RestURL'),
|
|
||||||
'UserName': config.get('storage', 'UserName'),
|
|
||||||
'Password': config.get('storage', 'Password'),
|
|
||||||
'StoragePool': config.get('storage', 'StoragePool')}
|
|
||||||
self.fusionstorage_conf._encode_authentication(storage_info)
|
|
||||||
name_node = storage_info.get('UserName')
|
|
||||||
pwd_node = storage_info.get('Password')
|
|
||||||
self.assertEqual('!&&&ZmFrZV91c2Vy', name_node)
|
|
||||||
self.assertEqual('!&&&ZmFrZV9wYXNzd2Q=', pwd_node)
|
|
||||||
|
|
||||||
def test__manager_ip(self):
|
def test__manager_ip(self):
|
||||||
manager_ips = {'fake_host': 'fake_ip'}
|
manager_ips = {'fake_host': 'fake_ip'}
|
||||||
self.mock_object(
|
self.mock_object(
|
||||||
|
@ -23,9 +23,8 @@ VOLUME_NOT_EXIST = 31000000
|
|||||||
BASIC_URI = '/dsware/service/'
|
BASIC_URI = '/dsware/service/'
|
||||||
CONF_PATH = "/etc/cinder/cinder.conf"
|
CONF_PATH = "/etc/cinder/cinder.conf"
|
||||||
|
|
||||||
CONF_ADDRESS = "RestURL"
|
CONF_ADDRESS = "dsware_rest_url"
|
||||||
CONF_MANAGER_IP = "manager_ips"
|
CONF_MANAGER_IP = "manager_ips"
|
||||||
CONF_POOLS = "StoragePool"
|
CONF_POOLS = "dsware_storage_pools"
|
||||||
CONF_PWD = "Password"
|
CONF_PWD = "san_password"
|
||||||
CONF_STORAGE = "storage"
|
CONF_USER = "san_login"
|
||||||
CONF_USER = "UserName"
|
|
||||||
|
@ -25,6 +25,7 @@ from cinder import interface
|
|||||||
from cinder.volume import driver
|
from cinder.volume import driver
|
||||||
from cinder.volume.drivers.fusionstorage import fs_client
|
from cinder.volume.drivers.fusionstorage import fs_client
|
||||||
from cinder.volume.drivers.fusionstorage import fs_conf
|
from cinder.volume.drivers.fusionstorage import fs_conf
|
||||||
|
from cinder.volume.drivers.san import san
|
||||||
from cinder.volume import utils as volume_utils
|
from cinder.volume import utils as volume_utils
|
||||||
|
|
||||||
LOG = logging.getLogger(__name__)
|
LOG = logging.getLogger(__name__)
|
||||||
@ -89,13 +90,15 @@ volume_opts = [
|
|||||||
help='This option is to support the FSA to mount across the '
|
help='This option is to support the FSA to mount across the '
|
||||||
'different nodes. The parameters takes the standard dict '
|
'different nodes. The parameters takes the standard dict '
|
||||||
'config form, manager_ips = host1:ip1, host2:ip2...'),
|
'config form, manager_ips = host1:ip1, host2:ip2...'),
|
||||||
cfg.DictOpt('storage',
|
cfg.StrOpt('dsware_rest_url',
|
||||||
default={},
|
default='',
|
||||||
secret=True,
|
help='The address of FusionStorage array. For example, '
|
||||||
help='This field is configured with the information of array '
|
'"dsware_rest_url=xxx"'),
|
||||||
'and user info. The parameters takes the standard dict '
|
cfg.StrOpt('dsware_storage_pools',
|
||||||
'config form, Storage = UserName:xxx, Password:xxx, '
|
default="",
|
||||||
'RestURL:xxx')
|
help='The list of pools on the FusionStorage array, the '
|
||||||
|
'semicolon(;) was used to split the storage pools, '
|
||||||
|
'"dsware_storage_pools = xxx1; xxx2; xxx3"')
|
||||||
]
|
]
|
||||||
|
|
||||||
CONF = cfg.CONF
|
CONF = cfg.CONF
|
||||||
@ -116,6 +119,7 @@ class DSWAREDriver(driver.VolumeDriver):
|
|||||||
raise exception.InvalidInput(reason=msg)
|
raise exception.InvalidInput(reason=msg)
|
||||||
|
|
||||||
self.configuration.append_config_values(volume_opts)
|
self.configuration.append_config_values(volume_opts)
|
||||||
|
self.configuration.append_config_values(san.san_opts)
|
||||||
self.conf = fs_conf.FusionStorageConf(self.configuration, self.host)
|
self.conf = fs_conf.FusionStorageConf(self.configuration, self.host)
|
||||||
self.client = None
|
self.client = None
|
||||||
|
|
||||||
|
@ -44,16 +44,15 @@ class FusionStorageConf(object):
|
|||||||
raise exception.InvalidInput(reason=msg)
|
raise exception.InvalidInput(reason=msg)
|
||||||
|
|
||||||
def update_config_value(self):
|
def update_config_value(self):
|
||||||
storage_info = self.configuration.safe_get(constants.CONF_STORAGE)
|
self._encode_authentication()
|
||||||
self._pools_name(storage_info)
|
self._pools_name()
|
||||||
self._san_address(storage_info)
|
self._san_address()
|
||||||
self._encode_authentication(storage_info)
|
self._san_user()
|
||||||
self._san_user(storage_info)
|
self._san_password()
|
||||||
self._san_password(storage_info)
|
|
||||||
|
|
||||||
def _encode_authentication(self, storage_info):
|
def _encode_authentication(self):
|
||||||
name_node = storage_info.get(constants.CONF_USER)
|
name_node = self.configuration.safe_get(constants.CONF_USER)
|
||||||
pwd_node = storage_info.get(constants.CONF_PWD)
|
pwd_node = self.configuration.safe_get(constants.CONF_PWD)
|
||||||
|
|
||||||
need_encode = False
|
need_encode = False
|
||||||
if name_node is not None and not name_node.startswith('!&&&'):
|
if name_node is not None and not name_node.startswith('!&&&'):
|
||||||
@ -67,29 +66,18 @@ class FusionStorageConf(object):
|
|||||||
need_encode = True
|
need_encode = True
|
||||||
|
|
||||||
if need_encode:
|
if need_encode:
|
||||||
self._rewrite_conf(storage_info, name_node, pwd_node)
|
self._rewrite_conf(name_node, pwd_node)
|
||||||
|
|
||||||
def _rewrite_conf(self, storage_info, name_node, pwd_node):
|
def _rewrite_conf(self, name_node, pwd_node):
|
||||||
storage_info.update({constants.CONF_USER: name_node,
|
|
||||||
constants.CONF_PWD: pwd_node})
|
|
||||||
storage_info = ("\n %(conf_name)s: %(name)s,"
|
|
||||||
"\n %(conf_pwd)s: %(pwd)s,"
|
|
||||||
"\n %(conf_url)s: %(url)s,"
|
|
||||||
"\n %(conf_pool)s: %(pool)s"
|
|
||||||
% {"conf_name": constants.CONF_USER,
|
|
||||||
"conf_pwd": constants.CONF_PWD,
|
|
||||||
"conf_url": constants.CONF_ADDRESS,
|
|
||||||
"conf_pool": constants.CONF_POOLS,
|
|
||||||
"name": name_node,
|
|
||||||
"pwd": pwd_node,
|
|
||||||
"url": storage_info.get(constants.CONF_ADDRESS),
|
|
||||||
"pool": storage_info.get(constants.CONF_POOLS)})
|
|
||||||
if os.path.exists(constants.CONF_PATH):
|
if os.path.exists(constants.CONF_PATH):
|
||||||
utils.execute("chmod", "666", constants.CONF_PATH,
|
utils.execute("chmod", "666", constants.CONF_PATH,
|
||||||
run_as_root=True)
|
run_as_root=True)
|
||||||
conf = configparser.ConfigParser()
|
conf = configparser.ConfigParser()
|
||||||
conf.read(constants.CONF_PATH)
|
conf.read(constants.CONF_PATH)
|
||||||
conf.set(self.host, constants.CONF_STORAGE, storage_info)
|
if name_node:
|
||||||
|
conf.set(self.host, constants.CONF_USER, name_node)
|
||||||
|
if pwd_node:
|
||||||
|
conf.set(self.host, constants.CONF_PWD, pwd_node)
|
||||||
fh = open(constants.CONF_PATH, 'w')
|
fh = open(constants.CONF_PATH, 'w')
|
||||||
conf.write(fh)
|
conf.write(fh)
|
||||||
fh.close()
|
fh.close()
|
||||||
@ -102,25 +90,29 @@ class FusionStorageConf(object):
|
|||||||
LOG.error(msg)
|
LOG.error(msg)
|
||||||
raise exception.InvalidInput(reason=msg)
|
raise exception.InvalidInput(reason=msg)
|
||||||
|
|
||||||
def _san_address(self, storage_info):
|
def _san_address(self):
|
||||||
address = storage_info.get(constants.CONF_ADDRESS)
|
address = self.configuration.safe_get(constants.CONF_ADDRESS)
|
||||||
self._assert_text_result(address, mess=constants.CONF_ADDRESS)
|
self._assert_text_result(address, mess=constants.CONF_ADDRESS)
|
||||||
setattr(self.configuration, 'san_address', address)
|
setattr(self.configuration, 'san_address', address)
|
||||||
|
|
||||||
def _san_user(self, storage_info):
|
def _decode_text(self, text):
|
||||||
user_text = storage_info.get(constants.CONF_USER)
|
return (base64.b64decode(six.b(text[4:])).decode() if
|
||||||
|
text.startswith('!&&&') else text)
|
||||||
|
|
||||||
|
def _san_user(self):
|
||||||
|
user_text = self.configuration.safe_get(constants.CONF_USER)
|
||||||
self._assert_text_result(user_text, mess=constants.CONF_USER)
|
self._assert_text_result(user_text, mess=constants.CONF_USER)
|
||||||
user = base64.b64decode(six.b(user_text[4:])).decode()
|
user = self._decode_text(user_text)
|
||||||
setattr(self.configuration, 'san_user', user)
|
setattr(self.configuration, 'san_user', user)
|
||||||
|
|
||||||
def _san_password(self, storage_info):
|
def _san_password(self):
|
||||||
pwd_text = storage_info.get(constants.CONF_PWD)
|
pwd_text = self.configuration.safe_get(constants.CONF_PWD)
|
||||||
self._assert_text_result(pwd_text, mess=constants.CONF_PWD)
|
self._assert_text_result(pwd_text, mess=constants.CONF_PWD)
|
||||||
pwd = base64.b64decode(six.b(pwd_text[4:])).decode()
|
pwd = self._decode_text(pwd_text)
|
||||||
setattr(self.configuration, 'san_password', pwd)
|
setattr(self.configuration, 'san_password', pwd)
|
||||||
|
|
||||||
def _pools_name(self, storage_info):
|
def _pools_name(self):
|
||||||
pools_name = storage_info.get(constants.CONF_POOLS)
|
pools_name = self.configuration.safe_get(constants.CONF_POOLS)
|
||||||
self._assert_text_result(pools_name, mess=constants.CONF_POOLS)
|
self._assert_text_result(pools_name, mess=constants.CONF_POOLS)
|
||||||
pools = set(x.strip() for x in pools_name.split(';') if x.strip())
|
pools = set(x.strip() for x in pools_name.split(';') if x.strip())
|
||||||
if not pools:
|
if not pools:
|
||||||
|
@ -0,0 +1,24 @@
|
|||||||
|
---
|
||||||
|
upgrades:
|
||||||
|
- The FusionStorage driver has added the configuration options "manager_ips",
|
||||||
|
"dsware_rest_url", "san_login", "san_password" and "dsware_storage_pools".
|
||||||
|
"[<backend_id>]/manager_ips", the ips of FusionStorage Agent(FSA). This
|
||||||
|
option is to support FSA to mount accross the different FSA nodes. The
|
||||||
|
parameters takes the standard dict config form, manager_ips = host1:ip1,
|
||||||
|
host2:ip2...
|
||||||
|
"[<backend_id>]/dsware_rest_url", the address and port of FusionStorage
|
||||||
|
Manager(FSM) in the format of a string. Currently, only one "dsware_rest_url"
|
||||||
|
is supported.
|
||||||
|
"[<backend_id>]/san_login", the user name of the FusionStorage Manager(FSM)
|
||||||
|
in the format of a string. Currently, only one "san_login" is supported.
|
||||||
|
"[<backend_id>]/san_password", the user password of FusionStorage Manager(FSM)
|
||||||
|
in the format of a string. Currently, only one "san_password" is supported.
|
||||||
|
"[<backend_id>]/dsware_storage_pools", the name of the storage pools that
|
||||||
|
exists on the FusionStorage Manager. Multiple storage pools can be configed,
|
||||||
|
separated by a semicolon.
|
||||||
|
deprecations:
|
||||||
|
- The FusionStorage driver has deprecated the configuration options
|
||||||
|
"dsware_isthin", "dsware_manager", "fusionstorageagent",
|
||||||
|
"clone_volume_timeout", "pool_type", and "pool_id_filter". These configuration
|
||||||
|
options will be removed in the Train release(14.0.0).
|
||||||
|
|
Loading…
Reference in New Issue
Block a user