Merge "Infrastructure to use the DRBD transport for NOVA"

This commit is contained in:
Jenkins 2016-01-16 12:23:58 +00:00 committed by Gerrit Code Review
commit a329fd9c29
3 changed files with 493 additions and 78 deletions

View File

@ -14,6 +14,8 @@
# under the License. # under the License.
import collections import collections
import six
import sys
import mock import mock
from oslo_utils import importutils from oslo_utils import importutils
@ -33,17 +35,37 @@ class mock_dbus(object):
return defaults return defaults
class mock_dm_utils(object): class mock_dm_consts(object):
@staticmethod
def dict_to_aux_props(x):
return x
class mock_dm_const(object):
TQ_GET_PATH = "get_path" TQ_GET_PATH = "get_path"
NODE_ADDR = "addr"
CSTATE_PREFIX = "cstate:"
TSTATE_PREFIX = "tstate:"
FLAG_UPD_POOL = "upd_pool"
FLAG_UPDATE = "update"
FLAG_DRBDCTRL = "drbdctrl"
FLAG_STORAGE = "storage"
FLAG_EXTERNAL = "external"
FLAG_DEPLOY = "deploy"
FLAG_DISKLESS = "diskless"
FLAG_CONNECT = "connect"
FLAG_UPD_CON = "upd_con"
FLAG_RECONNECT = "reconnect"
FLAG_OVERWRITE = "overwrite"
FLAG_DISCARD = "discard"
FLAG_UPD_CONFIG = "upd_config"
FLAG_STANDBY = "standby"
FLAG_QIGNORE = "qignore"
AUX_PROP_PREFIX = "aux:"
BOOL_TRUE = "true"
BOOL_FALSE = "true"
class mock_dm_exc(object): class mock_dm_exc(object):
@ -52,22 +74,54 @@ class mock_dm_exc(object):
DM_ENOENT = 2 DM_ENOENT = 2
DM_ERROR = 1000 DM_ERROR = 1000
pass
class mock_dm_utils(object):
@staticmethod
def _aux_prop_name(key):
if six.text_type(key).startswith(mock_dm_consts.AUX_PROP_PREFIX):
return key[len(mock_dm_consts.AUX_PROP_PREFIX):]
else:
return None
@staticmethod
def aux_props_to_dict(props):
aux_props = {}
for (key, val) in props.items():
aux_key = mock_dm_utils._aux_prop_name(key)
if aux_key is not None:
aux_props[aux_key] = val
return aux_props
@staticmethod
def dict_to_aux_props(props):
aux_props = {}
for (key, val) in props.items():
aux_key = mock_dm_consts.AUX_PROP_PREFIX + six.text_type(key)
aux_props[aux_key] = six.text_type(val)
return aux_props
def public_keys(c):
return [n for n in c.__dict__.keys() if not n.startswith("_")]
import sys
sys.modules['dbus'] = mock_dbus sys.modules['dbus'] = mock_dbus
sys.modules['drbdmanage'] = collections.namedtuple( sys.modules['drbdmanage'] = collections.namedtuple(
'module', ['consts', 'exceptions', 'utils']) 'module', ['consts', 'exceptions', 'utils'])
sys.modules['drbdmanage.utils'] = collections.namedtuple( sys.modules['drbdmanage.utils'] = collections.namedtuple(
'module', ['dict_to_aux_props']) 'module', public_keys(mock_dm_utils))
sys.modules['drbdmanage.consts'] = collections.namedtuple( sys.modules['drbdmanage.consts'] = collections.namedtuple(
'module', []) 'module', public_keys(mock_dm_consts))
sys.modules['drbdmanage.exceptions'] = collections.namedtuple( sys.modules['drbdmanage.exceptions'] = collections.namedtuple(
'module', ['DM_EEXIST']) 'module', public_keys(mock_dm_exc))
import cinder.volume.drivers.drbdmanagedrv as drv
from cinder.volume.drivers import drbdmanagedrv drv.dbus = mock_dbus
drv.dm_const = mock_dm_consts
drv.dm_utils = mock_dm_utils
drv.dm_exc = mock_dm_exc
def create_configuration(object): def create_configuration(object):
@ -86,7 +140,8 @@ class DrbdManageFakeDriver(object):
def list_resources(self, res, serial, prop, req): def list_resources(self, res, serial, prop, req):
self.calls.append(["list_resources", res, prop, req]) self.calls.append(["list_resources", res, prop, req])
if 'cinder-id' in prop and prop['cinder-id'].startswith("deadbeef"): if ('aux:cinder-id' in prop and
prop['aux:cinder-id'].startswith("deadbeef")):
return ([[mock_dm_exc.DM_ENOENT, "none", []]], return ([[mock_dm_exc.DM_ENOENT, "none", []]],
[]) [])
else: else:
@ -107,7 +162,8 @@ class DrbdManageFakeDriver(object):
def list_volumes(self, res, ser, prop, req): def list_volumes(self, res, ser, prop, req):
self.calls.append(["list_volumes", res, ser, prop, req]) self.calls.append(["list_volumes", res, ser, prop, req])
if 'cinder-id' in prop and prop['cinder-id'].startswith("deadbeef"): if ('aux:cinder-id' in prop and
prop['aux:cinder-id'].startswith("deadbeef")):
return ([[mock_dm_exc.DM_SUCCESS, "none", []]], return ([[mock_dm_exc.DM_SUCCESS, "none", []]],
[]) [])
else: else:
@ -121,13 +177,14 @@ class DrbdManageFakeDriver(object):
def text_query(self, cmd): def text_query(self, cmd):
self.calls.append(["text_query", cmd]) self.calls.append(["text_query", cmd])
if cmd[0] == mock_dm_const.TQ_GET_PATH: if cmd[0] == mock_dm_consts.TQ_GET_PATH:
return ([(mock_dm_exc.DM_SUCCESS, "ack", [])], ['/dev/drbd0']) return ([(mock_dm_exc.DM_SUCCESS, "ack", [])], ['/dev/drbd0'])
return ([(mock_dm_exc.DM_ERROR, 'unknown command', [])], []) return ([(mock_dm_exc.DM_ERROR, 'unknown command', [])], [])
def list_assignments(self, nodes, res, ser, prop, req): def list_assignments(self, nodes, res, ser, prop, req):
self.calls.append(["list_assignments", nodes, res, ser, prop, req]) self.calls.append(["list_assignments", nodes, res, ser, prop, req])
if 'cinder-id' in prop and prop['cinder-id'].startswith("deadbeef"): if ('aux:cinder-id' in prop and
prop['aux:cinder-id'].startswith("deadbeef")):
return ([[mock_dm_exc.DM_SUCCESS, "none", []]], return ([[mock_dm_exc.DM_SUCCESS, "none", []]],
[]) [])
else: else:
@ -141,7 +198,8 @@ class DrbdManageFakeDriver(object):
def list_snapshots(self, res, sn, serial, prop, req): def list_snapshots(self, res, sn, serial, prop, req):
self.calls.append(["list_snapshots", res, sn, serial, prop, req]) self.calls.append(["list_snapshots", res, sn, serial, prop, req])
if 'cinder-id' in prop and prop['cinder-id'].startswith("deadbeef"): if ('aux:cinder-id' in prop and
prop['aux:cinder-id'].startswith("deadbeef")):
return ([[mock_dm_exc.DM_SUCCESS, "none", []]], return ([[mock_dm_exc.DM_SUCCESS, "none", []]],
[]) [])
else: else:
@ -165,8 +223,15 @@ class DrbdManageFakeDriver(object):
self.calls.append(["assign", host, resource, props]) self.calls.append(["assign", host, resource, props])
return [[mock_dm_exc.DM_SUCCESS, "ack", []]] return [[mock_dm_exc.DM_SUCCESS, "ack", []]]
def create_node(self, name, prop):
self.calls.append(["create_node", name, prop])
if name.startswith('EXIST'):
return [(mock_dm_exc.DM_EEXIST, "none", [])]
else:
return [(mock_dm_exc.DM_SUCCESS, "ack", [])]
class DrbdManageTestCase(test.TestCase):
class DrbdManageIscsiTestCase(test.TestCase):
def setUp(self): def setUp(self):
self.ctxt = context.get_admin_context() self.ctxt = context.get_admin_context()
@ -175,23 +240,19 @@ class DrbdManageTestCase(test.TestCase):
self.configuration.san_is_local = True self.configuration.san_is_local = True
self.configuration.reserved_percentage = 1 self.configuration.reserved_percentage = 1
super(DrbdManageTestCase, self).setUp() super(DrbdManageIscsiTestCase, self).setUp()
self.stubs.Set(importutils, 'import_object', self.stubs.Set(importutils, 'import_object',
self.fake_import_object) self.fake_import_object)
self.stubs.Set(drbdmanagedrv.DrbdManageDriver, self.stubs.Set(drv.DrbdManageBaseDriver,
'call_or_reconnect', 'call_or_reconnect',
self.fake_issue_dbus_call) self.fake_issue_dbus_call)
self.stubs.Set(drbdmanagedrv.DrbdManageDriver, self.stubs.Set(drv.DrbdManageBaseDriver,
'dbus_connect', 'dbus_connect',
self.fake_issue_dbus_connect) self.fake_issue_dbus_connect)
self.stubs.Set(drv.DrbdManageBaseDriver,
sys.modules['cinder.volume.drivers.drbdmanagedrv'].dm_const \ '_wait_for_node_assignment',
= mock_dm_const self.fake_wait_node_assignment)
sys.modules['cinder.volume.drivers.drbdmanagedrv'].dm_utils \
= mock_dm_utils
sys.modules['cinder.volume.drivers.drbdmanagedrv'].dm_exc \
= mock_dm_exc
self.configuration.safe_get = lambda x: 'fake' self.configuration.safe_get = lambda x: 'fake'
@ -202,12 +263,18 @@ class DrbdManageTestCase(test.TestCase):
def fake_issue_dbus_call(self, fn, *args): def fake_issue_dbus_call(self, fn, *args):
return fn(*args) return fn(*args)
def fake_wait_node_assignment(self, *args, **kwargs):
return True
def fake_issue_dbus_connect(self): def fake_issue_dbus_connect(self):
self.odm = DrbdManageFakeDriver() self.odm = DrbdManageFakeDriver()
def call_or_reconnect(self, method, *params): def call_or_reconnect(self, method, *params):
return method(*params) return method(*params)
def fake_is_external_node(self, name):
return False
# Tests per se # Tests per se
def test_create_volume(self): def test_create_volume(self):
@ -218,7 +285,7 @@ class DrbdManageTestCase(test.TestCase):
'volume_type_id': 'drbdmanage', 'volume_type_id': 'drbdmanage',
'created_at': timeutils.utcnow()} 'created_at': timeutils.utcnow()}
dmd = drbdmanagedrv.DrbdManageDriver(configuration=self.configuration) dmd = drv.DrbdManageIscsiDriver(configuration=self.configuration)
dmd.drbdmanage_devs_on_controller = False dmd.drbdmanage_devs_on_controller = False
dmd.odm = DrbdManageFakeDriver() dmd.odm = DrbdManageFakeDriver()
dmd.create_volume(testvol) dmd.create_volume(testvol)
@ -237,7 +304,7 @@ class DrbdManageTestCase(test.TestCase):
'volume_type_id': 'drbdmanage', 'volume_type_id': 'drbdmanage',
'created_at': timeutils.utcnow()} 'created_at': timeutils.utcnow()}
dmd = drbdmanagedrv.DrbdManageDriver(configuration=self.configuration) dmd = drv.DrbdManageIscsiDriver(configuration=self.configuration)
dmd.drbdmanage_devs_on_controller = True dmd.drbdmanage_devs_on_controller = True
dmd.odm = DrbdManageFakeDriver() dmd.odm = DrbdManageFakeDriver()
dmd.create_volume(testvol) dmd.create_volume(testvol)
@ -257,11 +324,11 @@ class DrbdManageTestCase(test.TestCase):
'volume_type_id': 'drbdmanage', 'volume_type_id': 'drbdmanage',
'created_at': timeutils.utcnow()} 'created_at': timeutils.utcnow()}
dmd = drbdmanagedrv.DrbdManageDriver(configuration=self.configuration) dmd = drv.DrbdManageIscsiDriver(configuration=self.configuration)
dmd.odm = DrbdManageFakeDriver() dmd.odm = DrbdManageFakeDriver()
dmd.delete_volume(testvol) dmd.delete_volume(testvol)
self.assertEqual("list_volumes", dmd.odm.calls[0][0]) self.assertEqual("list_volumes", dmd.odm.calls[0][0])
self.assertEqual(testvol['id'], dmd.odm.calls[0][3]["cinder-id"]) self.assertEqual(testvol['id'], dmd.odm.calls[0][3]["aux:cinder-id"])
self.assertEqual("remove_volume", dmd.odm.calls[1][0]) self.assertEqual("remove_volume", dmd.odm.calls[1][0])
def test_local_path(self): def test_local_path(self):
@ -272,7 +339,7 @@ class DrbdManageTestCase(test.TestCase):
'volume_type_id': 'drbdmanage', 'volume_type_id': 'drbdmanage',
'created_at': timeutils.utcnow()} 'created_at': timeutils.utcnow()}
dmd = drbdmanagedrv.DrbdManageDriver(configuration=self.configuration) dmd = drv.DrbdManageIscsiDriver(configuration=self.configuration)
dmd.odm = DrbdManageFakeDriver() dmd.odm = DrbdManageFakeDriver()
data = dmd.local_path(testvol) data = dmd.local_path(testvol)
self.assertTrue(data.startswith("/dev/drbd")) self.assertTrue(data.startswith("/dev/drbd"))
@ -281,7 +348,7 @@ class DrbdManageTestCase(test.TestCase):
testsnap = {'id': 'ca253fd0-8068-11e4-98c0-5254008ea111', testsnap = {'id': 'ca253fd0-8068-11e4-98c0-5254008ea111',
'volume_id': 'ba253fd0-8068-11e4-98c0-5254008ea111'} 'volume_id': 'ba253fd0-8068-11e4-98c0-5254008ea111'}
dmd = drbdmanagedrv.DrbdManageDriver(configuration=self.configuration) dmd = drv.DrbdManageIscsiDriver(configuration=self.configuration)
dmd.odm = DrbdManageFakeDriver() dmd.odm = DrbdManageFakeDriver()
dmd.create_snapshot(testsnap) dmd.create_snapshot(testsnap)
self.assertEqual("list_volumes", dmd.odm.calls[0][0]) self.assertEqual("list_volumes", dmd.odm.calls[0][0])
@ -292,7 +359,7 @@ class DrbdManageTestCase(test.TestCase):
def test_delete_snapshot(self): def test_delete_snapshot(self):
testsnap = {'id': 'ca253fd0-8068-11e4-98c0-5254008ea111'} testsnap = {'id': 'ca253fd0-8068-11e4-98c0-5254008ea111'}
dmd = drbdmanagedrv.DrbdManageDriver(configuration=self.configuration) dmd = drv.DrbdManageIscsiDriver(configuration=self.configuration)
dmd.odm = DrbdManageFakeDriver() dmd.odm = DrbdManageFakeDriver()
dmd.delete_snapshot(testsnap) dmd.delete_snapshot(testsnap)
self.assertEqual("list_snapshots", dmd.odm.calls[0][0]) self.assertEqual("list_snapshots", dmd.odm.calls[0][0])
@ -306,11 +373,11 @@ class DrbdManageTestCase(test.TestCase):
'volume_type_id': 'drbdmanage', 'volume_type_id': 'drbdmanage',
'created_at': timeutils.utcnow()} 'created_at': timeutils.utcnow()}
dmd = drbdmanagedrv.DrbdManageDriver(configuration=self.configuration) dmd = drv.DrbdManageIscsiDriver(configuration=self.configuration)
dmd.odm = DrbdManageFakeDriver() dmd.odm = DrbdManageFakeDriver()
dmd.extend_volume(testvol, 5) dmd.extend_volume(testvol, 5)
self.assertEqual("list_volumes", dmd.odm.calls[0][0]) self.assertEqual("list_volumes", dmd.odm.calls[0][0])
self.assertEqual(testvol['id'], dmd.odm.calls[0][3]["cinder-id"]) self.assertEqual(testvol['id'], dmd.odm.calls[0][3]["aux:cinder-id"])
self.assertEqual("resize_volume", dmd.odm.calls[1][0]) self.assertEqual("resize_volume", dmd.odm.calls[1][0])
self.assertEqual("res", dmd.odm.calls[1][1]) self.assertEqual("res", dmd.odm.calls[1][1])
self.assertEqual(2, dmd.odm.calls[1][2]) self.assertEqual(2, dmd.odm.calls[1][2])
@ -327,7 +394,7 @@ class DrbdManageTestCase(test.TestCase):
newvol = {'id': 'ca253fd0-8068-11e4-98c0-5254008ea111'} newvol = {'id': 'ca253fd0-8068-11e4-98c0-5254008ea111'}
dmd = drbdmanagedrv.DrbdManageDriver(configuration=self.configuration) dmd = drv.DrbdManageIscsiDriver(configuration=self.configuration)
dmd.odm = DrbdManageFakeDriver() dmd.odm = DrbdManageFakeDriver()
dmd.create_cloned_volume(newvol, srcvol) dmd.create_cloned_volume(newvol, srcvol)
self.assertEqual("list_volumes", dmd.odm.calls[0][0]) self.assertEqual("list_volumes", dmd.odm.calls[0][0])
@ -338,3 +405,36 @@ class DrbdManageTestCase(test.TestCase):
self.assertEqual("list_snapshots", dmd.odm.calls[5][0]) self.assertEqual("list_snapshots", dmd.odm.calls[5][0])
self.assertEqual("remove_snapshot", dmd.odm.calls[6][0]) self.assertEqual("remove_snapshot", dmd.odm.calls[6][0])
self.assertEqual("remove_snapshot", dmd.odm.calls[6][0]) self.assertEqual("remove_snapshot", dmd.odm.calls[6][0])
class DrbdManageDrbdTestCase(DrbdManageIscsiTestCase):
def setUp(self):
super(DrbdManageDrbdTestCase, self).setUp()
self.stubs.Set(drv.DrbdManageDrbdDriver,
'_is_external_node',
self.fake_is_external_node)
def test_drbd_create_export(self):
volume = {'project_id': 'testprjid',
'name': 'testvol',
'size': 1,
'id': 'ba253fd0-8068-11e4-98c0-5254008ea111',
'volume_type_id': 'drbdmanage',
'created_at': timeutils.utcnow()}
connector = {'host': 'node99',
'ip': '127.0.0.99'}
dmd = drv.DrbdManageDrbdDriver(configuration=self.configuration)
dmd.odm = DrbdManageFakeDriver()
x = dmd.create_export({}, volume, connector)
self.assertEqual("list_volumes", dmd.odm.calls[0][0])
self.assertEqual("create_node", dmd.odm.calls[1][0])
self.assertEqual("assign", dmd.odm.calls[2][0])
# local_path
self.assertEqual("list_volumes", dmd.odm.calls[3][0])
self.assertEqual("text_query", dmd.odm.calls[4][0])
self.assertEqual("local", x["driver_volume_type"])

View File

@ -17,14 +17,16 @@
""" """
This driver connects Cinder to an installed DRBDmanage instance, see This driver connects Cinder to an installed DRBDmanage instance, see
http://oss.linbit.com/drbdmanage/ http://drbd.linbit.com/users-guide-9.0/ch-openstack.html
http://git.linbit.com/drbdmanage.git/
for more details. for more details.
""" """
import eventlet
import six import six
import socket import socket
import time
import uuid import uuid
from oslo_config import cfg from oslo_config import cfg
@ -34,7 +36,7 @@ from oslo_utils import units
from cinder import exception from cinder import exception
from cinder.i18n import _, _LW, _LI from cinder.i18n import _, _LW, _LI, _LE
from cinder.volume import driver from cinder.volume import driver
try: try:
@ -43,6 +45,7 @@ try:
import drbdmanage.exceptions as dm_exc import drbdmanage.exceptions as dm_exc
import drbdmanage.utils as dm_utils import drbdmanage.utils as dm_utils
except ImportError: except ImportError:
# Used for the tests, when no DRBDmanage is installed
dbus = None dbus = None
dm_const = None dm_const = None
dm_exc = None dm_exc = None
@ -52,8 +55,8 @@ except ImportError:
LOG = logging.getLogger(__name__) LOG = logging.getLogger(__name__)
drbd_opts = [ drbd_opts = [
cfg.StrOpt('drbdmanage_redundancy', cfg.IntOpt('drbdmanage_redundancy',
default='1', default=1,
help='Number of nodes that should replicate the data.'), help='Number of nodes that should replicate the data.'),
cfg.BoolOpt('drbdmanage_devs_on_controller', cfg.BoolOpt('drbdmanage_devs_on_controller',
default=True, default=True,
@ -72,21 +75,30 @@ CONF.register_opts(drbd_opts)
AUX_PROP_CINDER_VOL_ID = "cinder-id" AUX_PROP_CINDER_VOL_ID = "cinder-id"
AUX_PROP_TEMP_CLIENT = "cinder-is-temp-client"
DM_VN_PREFIX = 'CV_' # sadly 2CV isn't allowed by DRBDmanage DM_VN_PREFIX = 'CV_' # sadly 2CV isn't allowed by DRBDmanage
DM_SN_PREFIX = 'SN_' DM_SN_PREFIX = 'SN_'
class DrbdManageDriver(driver.VolumeDriver): # Need to be set later, so that the tests can fake
CS_DEPLOYED = None
CS_DISKLESS = None
CS_UPD_CON = None
class DrbdManageBaseDriver(driver.VolumeDriver):
"""Cinder driver that uses DRBDmanage for storage.""" """Cinder driver that uses DRBDmanage for storage."""
VERSION = '1.0.0' VERSION = '1.1.0'
drbdmanage_dbus_name = 'org.drbd.drbdmanaged' drbdmanage_dbus_name = 'org.drbd.drbdmanaged'
drbdmanage_dbus_interface = '/interface' drbdmanage_dbus_interface = '/interface'
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
self.empty_list = dbus.Array([], signature="a(s)") self.empty_list = dbus.Array([], signature="a(s)")
self.empty_dict = dbus.Array([], signature="a(ss)") self.empty_dict = dbus.Array([], signature="a(ss)")
super(DrbdManageDriver, self).__init__(*args, **kwargs)
super(DrbdManageBaseDriver, self).__init__(*args, **kwargs)
self.configuration.append_config_values(drbd_opts) self.configuration.append_config_values(drbd_opts)
if not self.drbdmanage_dbus_name: if not self.drbdmanage_dbus_name:
self.drbdmanage_dbus_name = 'org.drbd.drbdmanaged' self.drbdmanage_dbus_name = 'org.drbd.drbdmanaged'
@ -100,20 +112,15 @@ class DrbdManageDriver(driver.VolumeDriver):
True)) True))
self.dm_control_vol = ".drbdctrl" self.dm_control_vol = ".drbdctrl"
# Copied from the LVM driver, see self.backend_name = self.configuration.safe_get(
# I43190d1dac33748fe55fa00f260f32ab209be656 'volume_backend_name') or 'drbdmanage'
target_driver = self.target_mapping[
self.configuration.safe_get('iscsi_helper')]
LOG.debug('Attempting to initialize DRBD driver with the ' # needed as per pep8:
'following target_driver: %s', # F841 local variable 'CS_DEPLOYED' is assigned to but never used
target_driver) global CS_DEPLOYED, CS_DISKLESS, CS_UPD_CON
CS_DEPLOYED = dm_const.CSTATE_PREFIX + dm_const.FLAG_DEPLOY
self.target_driver = importutils.import_object( CS_DISKLESS = dm_const.CSTATE_PREFIX + dm_const.FLAG_DISKLESS
target_driver, CS_UPD_CON = dm_const.CSTATE_PREFIX + dm_const.FLAG_UPD_CON
configuration=self.configuration,
db=self.db,
executor=self._execute)
def dbus_connect(self): def dbus_connect(self):
self.odm = dbus.SystemBus().get_object(self.drbdmanage_dbus_name, self.odm = dbus.SystemBus().get_object(self.drbdmanage_dbus_name,
@ -132,7 +139,7 @@ class DrbdManageDriver(driver.VolumeDriver):
def do_setup(self, context): def do_setup(self, context):
"""Any initialization the volume driver does while starting.""" """Any initialization the volume driver does while starting."""
super(DrbdManageDriver, self).do_setup(context) super(DrbdManageBaseDriver, self).do_setup(context)
self.dbus_connect() self.dbus_connect()
def check_for_setup_error(self): def check_for_setup_error(self):
@ -199,6 +206,48 @@ class DrbdManageDriver(driver.VolumeDriver):
except ValueError: except ValueError:
return None return None
def _wait_for_node_assignment(self, res_name, vol_nr, nodenames,
filter_props=None, timeout=90,
check_vol_deployed=True):
"""Return True as soon as one assignment matches the filter."""
if not filter_props:
filter_props = self.empty_dict
end_time = time.time() + timeout
retry = 0
while time.time() < end_time:
res, assgs = self.call_or_reconnect(self.odm.list_assignments,
nodenames, [res_name], 0,
filter_props, self.empty_list)
self._check_result(res)
if len(assgs) > 0:
for assg in assgs:
vols = assg[3]
for v_nr, v_prop in vols:
if (v_nr == vol_nr):
if not check_vol_deployed:
# no need to check
return True
if v_prop[CS_DEPLOYED] == dm_const.BOOL_TRUE:
return True
retry += 1
# Not yet
LOG.warning(_LW('Try #%(try)d: Volume "%(res)s"/%(vol)d '
'not yet deployed on "%(host)s", waiting.'),
{'try': retry, 'host': nodenames,
'res': res_name, 'vol': vol_nr})
eventlet.sleep(min(0.5 + retry / 5, 2))
# Timeout
return False
def _priv_hash_from_volume(self, volume): def _priv_hash_from_volume(self, volume):
return dm_utils.dict_to_aux_props({ return dm_utils.dict_to_aux_props({
AUX_PROP_CINDER_VOL_ID: volume['id'], AUX_PROP_CINDER_VOL_ID: volume['id'],
@ -254,7 +303,7 @@ class DrbdManageDriver(driver.VolumeDriver):
LOG.debug("volume %(uuid)s is %(res)s/%(nr)d; %(rprop)s, %(vprop)s", LOG.debug("volume %(uuid)s is %(res)s/%(nr)d; %(rprop)s, %(vprop)s",
{'uuid': v_uuid, 'res': r_name, 'nr': v_nr, {'uuid': v_uuid, 'res': r_name, 'nr': v_nr,
'rprop': r_props, 'vprop': v_props}) 'rprop': dict(r_props), 'vprop': dict(v_props)})
return r_name, v_nr, r_props, v_props return r_name, v_nr, r_props, v_props
@ -295,7 +344,7 @@ class DrbdManageDriver(driver.VolumeDriver):
return r_name, s_name, s_props return r_name, s_name, s_props
def _resource_name_volnr_for_volume(self, volume, empty_ok=False): def _resource_name_volnr_for_volume(self, volume, empty_ok=False):
res, vol, _, _ = self._res_and_vl_data_for_volume(volume, empty_ok) res, vol, __, __ = self._res_and_vl_data_for_volume(volume, empty_ok)
return res, vol return res, vol
def local_path(self, volume): def local_path(self, volume):
@ -306,8 +355,10 @@ class DrbdManageDriver(driver.VolumeDriver):
dres, dres,
str(dvol)]) str(dvol)])
self._check_result(res) self._check_result(res)
if len(data) == 1: if len(data) == 1:
return data[0] return data[0]
message = _('Got bad path information from DRBDmanage! (%s)') % data message = _('Got bad path information from DRBDmanage! (%s)') % data
raise exception.VolumeBackendAPIException(data=message) raise exception.VolumeBackendAPIException(data=message)
@ -330,8 +381,8 @@ class DrbdManageDriver(driver.VolumeDriver):
# deploy gave an error on a previous try (like ENOSPC). # deploy gave an error on a previous try (like ENOSPC).
# Still, there might or might not be the volume in the resource - # Still, there might or might not be the volume in the resource -
# we have to check that explicitly. # we have to check that explicitly.
(_, drbd_vol) = self._resource_name_volnr_for_volume(volume, (__, drbd_vol) = self._resource_name_volnr_for_volume(volume,
empty_ok=True) empty_ok=True)
if not drbd_vol: if not drbd_vol:
props = self._priv_hash_from_volume(volume) props = self._priv_hash_from_volume(volume)
# TODO(PM): properties - redundancy, etc # TODO(PM): properties - redundancy, etc
@ -348,15 +399,18 @@ class DrbdManageDriver(driver.VolumeDriver):
0, True) 0, True)
self._check_result(res) self._check_result(res)
# TODO(pm): CG
self._wait_for_node_assignment(dres, 0, self.empty_list)
if self.drbdmanage_devs_on_controller: if self.drbdmanage_devs_on_controller:
# FIXME: Consistency groups, vol# # TODO(pm): CG
res = self.call_or_reconnect(self.odm.assign, res = self.call_or_reconnect(self.odm.assign,
socket.gethostname(), socket.gethostname(),
dres, dres,
self.empty_dict) self.empty_dict)
self._check_result(res, ignore=[dm_exc.DM_EEXIST]) self._check_result(res, ignore=[dm_exc.DM_EEXIST])
return 0 return {}
def delete_volume(self, volume): def delete_volume(self, volume):
"""Deletes a resource.""" """Deletes a resource."""
@ -431,20 +485,30 @@ class DrbdManageDriver(driver.VolumeDriver):
data["vendor_name"] = 'Open Source' data["vendor_name"] = 'Open Source'
data["driver_version"] = self.VERSION data["driver_version"] = self.VERSION
data["storage_protocol"] = self.target_driver.protocol
# This has to match the name set in the cinder volume driver spec, # This has to match the name set in the cinder volume driver spec,
# so keep it lowercase # so keep it lowercase
data["volume_backend_name"] = "drbdmanage" data["volume_backend_name"] = self.backend_name
data["pools"] = [] data["pools"] = []
res, free, total = self.call_or_reconnect(self.odm.cluster_free_query, res, free, total = self.call_or_reconnect(self.odm.cluster_free_query,
self.drbdmanage_redundancy) self.drbdmanage_redundancy)
self._check_result(res) self._check_result(res)
location_info = ('DrbdManageDriver:%(cvol)s:%(dbus)s' % location_info = ('DrbdManageBaseDriver:%(cvol)s:%(dbus)s' %
{'cvol': self.dm_control_vol, {'cvol': self.dm_control_vol,
'dbus': self.drbdmanage_dbus_name}) 'dbus': self.drbdmanage_dbus_name})
# add volumes
res, rl = self.call_or_reconnect(self.odm.list_volumes,
self.empty_list,
0,
self.empty_dict,
self.empty_list)
self._check_result(res)
total_volumes = 0
for res in rl:
total_volumes += len(res[2])
# TODO(PM): multiple DRBDmanage instances and/or multiple pools # TODO(PM): multiple DRBDmanage instances and/or multiple pools
single_pool = {} single_pool = {}
single_pool.update(dict( single_pool.update(dict(
@ -453,16 +517,14 @@ class DrbdManageDriver(driver.VolumeDriver):
total_capacity_gb=self._vol_size_to_cinder(total), total_capacity_gb=self._vol_size_to_cinder(total),
reserved_percentage=self.configuration.reserved_percentage, reserved_percentage=self.configuration.reserved_percentage,
location_info=location_info, location_info=location_info,
total_volumes=total_volumes,
filter_function=self.get_filter_function(),
goodness_function=self.get_goodness_function(),
QoS_support=False)) QoS_support=False))
data["pools"].append(single_pool) data["pools"].append(single_pool)
self._stats = data self._stats = data
def get_volume_stats(self, refresh=True):
"""Get volume status."""
self._update_volume_stats()
return self._stats return self._stats
def extend_volume(self, volume, new_size): def extend_volume(self, volume, new_size):
@ -487,7 +549,7 @@ class DrbdManageDriver(driver.VolumeDriver):
[dres], [dres],
0, 0,
self.empty_dict, self.empty_dict,
self.empty_dict) self.empty_list)
self._check_result(res) self._check_result(res)
nodes = [d[0] for d in data] nodes = [d[0] for d in data]
@ -518,7 +580,33 @@ class DrbdManageDriver(driver.VolumeDriver):
dres, sname, True) dres, sname, True)
return self._check_result(res, ignore=[dm_exc.DM_ENOENT]) return self._check_result(res, ignore=[dm_exc.DM_ENOENT])
# ####### Interface methods for DataPath (Target Driver) ########
# Class with iSCSI interface methods
class DrbdManageIscsiDriver(DrbdManageBaseDriver):
"""Cinder driver that uses the iSCSI protocol. """
def __init__(self, *args, **kwargs):
super(DrbdManageIscsiDriver, self).__init__(*args, **kwargs)
target_driver = self.target_mapping[
self.configuration.safe_get('iscsi_helper')]
LOG.debug('Attempting to initialize DRBD driver with the '
'following target_driver: %s',
target_driver)
self.target_driver = importutils.import_object(
target_driver,
configuration=self.configuration,
db=self.db,
executor=self._execute)
def get_volume_stats(self, refresh=False):
"""Get volume status."""
self._update_volume_stats()
self._stats["storage_protocol"] = "iSCSI"
return self._stats
def ensure_export(self, context, volume): def ensure_export(self, context, volume):
volume_path = self.local_path(volume) volume_path = self.local_path(volume)
@ -547,4 +635,230 @@ class DrbdManageDriver(driver.VolumeDriver):
return self.target_driver.validate_connector(connector) return self.target_driver.validate_connector(connector)
def terminate_connection(self, volume, connector, **kwargs): def terminate_connection(self, volume, connector, **kwargs):
return self.target_driver.terminate_connection(volume,
connector,
**kwargs)
return None return None
# for backwards compatibility keep the old class name, too
DrbdManageDriver = DrbdManageIscsiDriver
# Class with DRBD transport mode
class DrbdManageDrbdDriver(DrbdManageBaseDriver):
"""Cinder driver that uses the DRBD protocol. """
def __init__(self, *args, **kwargs):
super(DrbdManageDrbdDriver, self).__init__(*args, **kwargs)
def get_volume_stats(self, refresh=False):
"""Get volume status."""
self._update_volume_stats()
self._stats["storage_protocol"] = "DRBD"
return self._stats
def _return_local_access(self, nodename, volume,
dres=None, volume_path=None):
if not volume_path:
volume_path = self.local_path(volume)
return {
'driver_volume_type': 'local',
'data': {
"device_path": volume_path
}
}
def _return_drbdadm_config(self, volume, nodename,
dres=None, volume_path=None):
if not dres:
dres, dvol = self._resource_name_volnr_for_volume(volume)
res, data = self.call_or_reconnect(
self.odm.text_query,
['export_conf_split_up', nodename, dres])
self._check_result(res)
config = six.text_type(data.pop(0))
subst_data = {}
while len(data):
k = data.pop(0)
subst_data[k] = data.pop(0)
if not volume_path:
volume_path = self.local_path(volume)
return {
'driver_volume_type': 'drbd',
'data': {
'provider_location': ' '.join('drbd', nodename),
'device': volume_path,
# TODO(pm): consistency groups
'devices': [volume_path],
'provider_auth': subst_data['shared-secret'],
'config': config,
'name': dres,
}
}
def _is_external_node(self, nodename):
"""Return whether the given node is an "external" node."""
# If the node accessing the data (the "initiator" in iSCSI speak,
# "client" or "target" otherwise) is marked as an FLAG_EXTERNAL
# node, it does not have DRBDmanage active - and that means
# we have to send the necessary DRBD configuration.
#
# If DRBDmanage is running there, just pushing the (client)
# assignment is enough to make the local path available.
res, nodes = self.call_or_reconnect(self.odm.list_nodes,
[nodename], 0,
self.empty_dict,
[dm_const.FLAG_EXTERNAL])
self._check_result(res)
if len(nodes) != 1:
msg = _('Expected exactly one node called "%s"') % nodename
LOG.error(msg)
raise exception.VolumeDriverException(message=msg)
__, nodeattr = nodes[0]
return getattr(nodeattr, dm_const.FLAG_EXTERNAL,
dm_const.BOOL_FALSE) == dm_const.BOOL_TRUE
def _return_connection_data(self, nodename, volume, dres=None):
if self._is_external_node(nodename):
return self._return_drbdadm_config(nodename, volume, dres=dres)
else:
return self._return_local_access(nodename, volume)
def create_export(self, context, volume, connector):
dres, dvol = self._resource_name_volnr_for_volume(volume)
nodename = connector["host"]
# Ensure the node is known to DRBDmanage.
# Note that this does *not* mean that DRBDmanage has to
# be installed on it!
# This is just so that DRBD allows the IP to connect.
node_prop = {
dm_const.NODE_ADDR: connector["ip"],
dm_const.FLAG_DRBDCTRL: dm_const.BOOL_FALSE,
dm_const.FLAG_STORAGE: dm_const.BOOL_FALSE,
dm_const.FLAG_EXTERNAL: dm_const.BOOL_TRUE,
}
res = self.call_or_reconnect(
self.odm.create_node, nodename, node_prop)
self._check_result(res, ignore=[dm_exc.DM_EEXIST])
# Ensure the data is accessible, by creating an assignment.
assg_prop = {
dm_const.FLAG_DISKLESS: dm_const.BOOL_TRUE,
}
# If we create the assignment here, it's temporary -
# and has to be removed later on again.
assg_prop.update(dm_utils.aux_props_to_dict({
AUX_PROP_TEMP_CLIENT: dm_const.BOOL_TRUE,
}))
res = self.call_or_reconnect(
self.odm.assign, nodename, dres, assg_prop)
self._check_result(res, ignore=[dm_exc.DM_EEXIST])
# Wait for DRBDmanage to have completed that action.
# A DRBDmanage controlled node will set the cstate:deploy flag;
# an external node will not be available to change it, so we have
# to wait for the storage nodes to remove the upd_con flag
# (ie. they're now ready to receive the connection).
if self._is_external_node(nodename):
self._wait_for_node_assignment(
dres, dvol, [],
check_vol_deployed=False,
filter_props={
# must be deployed
CS_DEPLOYED: dm_const.BOOL_TRUE,
# must be a storage node (not diskless),
CS_DISKLESS: dm_const.BOOL_FALSE,
# connection must be available, no need for updating
CS_UPD_CON: dm_const.BOOL_FALSE,
})
else:
self._wait_for_node_assignment(
dres, dvol, [nodename],
check_vol_deployed=True,
filter_props={
CS_DEPLOYED: dm_const.BOOL_TRUE,
})
return self._return_connection_data(nodename, volume)
def ensure_export(self, context, volume):
fields = context['provider_location'].split(" ")
nodename = fields[1]
return self._return_connection_data(nodename, volume)
def initialize_connection(self, volume, connector):
nodename = connector["host"]
return self._return_connection_data(nodename, volume)
def terminate_connection(self, volume, connector,
force=False, **kwargs):
dres, dvol = self._resource_name_volnr_for_volume(
volume, empty_ok=True)
if not dres:
return
nodename = connector["host"]
# If the DRBD volume is diskless on that node, we remove it;
# if it has local storage, we keep it.
res, data = self.call_or_reconnect(
self.odm.list_assignments,
[nodename], [dres], 0,
self.empty_list, self.empty_list)
self._check_result(res, ignore=[dm_exc.DM_ENOENT])
if len(data) < 1:
# already removed?!
LOG.info(_LI('DRBD connection for %s already removed'),
volume['id'])
elif len(data) == 1:
__, __, props, __ = data[0]
my_props = dm_utils.dict_to_aux_props(props)
diskless = getattr(props,
dm_const.FLAG_DISKLESS,
dm_const.BOOL_FALSE)
temp_cli = getattr(my_props,
AUX_PROP_TEMP_CLIENT,
dm_const.BOOL_FALSE)
# If diskless assigned,
if ((diskless == dm_const.BOOL_TRUE) and
(temp_cli == dm_const.BOOL_TRUE)):
# remove the assignment
# TODO(pm): does it make sense to relay "force" here?
# What are the semantics?
# TODO(pm): consistency groups shouldn't really
# remove until *all* volumes are detached
res = self.call_or_reconnect(self.odm.unassign,
nodename, dres, force)
self._check_result(res, ignore=[dm_exc.DM_ENOENT])
else:
# more than one assignment?
LOG.error(_LE("DRBDmanage: too many assignments returned."))
return
def remove_export(self, context, volume):
pass

View File

@ -27,6 +27,7 @@ lvdisplay_lvmconf: EnvFilter, env, root, LVM_SYSTEM_DIR=, LC_ALL=C, lvdisplay
# os-brick.filters file instead and clean out stale brick values from # os-brick.filters file instead and clean out stale brick values from
# this file. # this file.
scsi_id: CommandFilter, /lib/udev/scsi_id, root scsi_id: CommandFilter, /lib/udev/scsi_id, root
drbdadm: CommandFilter, drbdadm, root
# cinder/brick/local_dev/lvm.py: 'vgcreate', vg_name, pv_list # cinder/brick/local_dev/lvm.py: 'vgcreate', vg_name, pv_list
vgcreate: CommandFilter, vgcreate, root vgcreate: CommandFilter, vgcreate, root