LVM driver: list manageable volumes and snapshots

Implement the list manageable volumes/snapshots APIs for the LVM
driver.

Change-Id: I5714449d66088b899aaa7c79c7ea15d32fe32ab1
Implements: blueprint list-manage-existing
This commit is contained in:
Avishay Traeger 2016-05-22 14:30:11 +03:00
parent af7bfedc51
commit 697b98b42f
6 changed files with 365 additions and 41 deletions

View File

@ -703,6 +703,44 @@ class LVM(executor.Executor):
return True return True
return False return False
def lv_is_snapshot(self, name):
"""Return True if LV is a snapshot, False otherwise."""
cmd = LVM.LVM_CMD_PREFIX + ['lvdisplay', '--noheading', '-C', '-o',
'Attr', '%s/%s' % (self.vg_name, name)]
out, _err = self._execute(*cmd,
root_helper=self._root_helper,
run_as_root=True)
out = out.strip()
if out:
if (out[0] == 's'):
return True
return False
def lv_is_open(self, name):
"""Return True if LV is currently open, False otherwise."""
cmd = LVM.LVM_CMD_PREFIX + ['lvdisplay', '--noheading', '-C', '-o',
'Attr', '%s/%s' % (self.vg_name, name)]
out, _err = self._execute(*cmd,
root_helper=self._root_helper,
run_as_root=True)
out = out.strip()
if out:
if (out[5] == 'o'):
return True
return False
def lv_get_origin(self, name):
"""Return the origin of an LV that is a snapshot, None otherwise."""
cmd = LVM.LVM_CMD_PREFIX + ['lvdisplay', '--noheading', '-C', '-o',
'Origin', '%s/%s' % (self.vg_name, name)]
out, _err = self._execute(*cmd,
root_helper=self._root_helper,
run_as_root=True)
out = out.strip()
if out:
return out
return None
def extend_volume(self, lv_name, new_size): def extend_volume(self, lv_name, new_size):
"""Extend the size of an existing volume.""" """Extend the size of an existing volume."""
# Volumes with snaps have attributes 'o' or 'O' and will be # Volumes with snaps have attributes 'o' or 'O' and will be

View File

@ -47,7 +47,7 @@ class BrickLvmTestCase(test.TestCase):
def fake_customised_lvm_version(obj, *cmd, **kwargs): def fake_customised_lvm_version(obj, *cmd, **kwargs):
return (" LVM version: 2.02.100(2)-RHEL6 (2013-09-12)\n", "") return (" LVM version: 2.02.100(2)-RHEL6 (2013-09-12)\n", "")
def fake_execute(obj, *cmd, **kwargs): def fake_execute(obj, *cmd, **kwargs): # noqa
cmd_string = ', '.join(cmd) cmd_string = ', '.join(cmd)
data = "\n" data = "\n"
@ -115,8 +115,18 @@ class BrickLvmTestCase(test.TestCase):
cmd_string): cmd_string):
if 'test-volumes' in cmd_string: if 'test-volumes' in cmd_string:
data = ' wi-a-' data = ' wi-a-'
elif 'snapshot' in cmd_string:
data = ' swi-a-s--'
elif 'open' in cmd_string:
data = ' -wi-ao---'
else: else:
data = ' owi-a-' data = ' owi-a-'
elif ('env, LC_ALL=C, lvdisplay, --noheading, -C, -o, Origin' in
cmd_string):
if 'snapshot' in cmd_string:
data = ' fake-volume-1'
else:
data = ' '
elif 'env, LC_ALL=C, pvs, --noheadings' in cmd_string: elif 'env, LC_ALL=C, pvs, --noheadings' in cmd_string:
data = " fake-vg|/dev/sda|10.00|1.00\n" data = " fake-vg|/dev/sda|10.00|1.00\n"
data += " fake-vg|/dev/sdb|10.00|1.00\n" data += " fake-vg|/dev/sdb|10.00|1.00\n"
@ -322,6 +332,19 @@ class BrickLvmTestCase(test.TestCase):
self.assertTrue(self.vg.lv_has_snapshot('fake-vg')) self.assertTrue(self.vg.lv_has_snapshot('fake-vg'))
self.assertFalse(self.vg.lv_has_snapshot('test-volumes')) self.assertFalse(self.vg.lv_has_snapshot('test-volumes'))
def test_lv_is_snapshot(self):
self.assertTrue(self.vg.lv_is_snapshot('fake-snapshot'))
self.assertFalse(self.vg.lv_is_snapshot('test-volumes'))
def test_lv_is_open(self):
self.assertTrue(self.vg.lv_is_open('fake-open'))
self.assertFalse(self.vg.lv_is_open('fake-snapshot'))
def test_lv_get_origin(self):
self.assertEqual('fake-volume-1',
self.vg.lv_get_origin('fake-snapshot'))
self.assertFalse(None, self.vg.lv_get_origin('test-volumes'))
def test_activate_lv(self): def test_activate_lv(self):
with mock.patch.object(self.vg, '_execute'): with mock.patch.object(self.vg, '_execute'):
self.vg._supports_lvchange_ignoreskipactivation = True self.vg._supports_lvchange_ignoreskipactivation = True

View File

@ -758,52 +758,116 @@ class VolumeUtilsTestCase(test.TestCase):
host_2 = 'fake_host2@backend1' host_2 = 'fake_host2@backend1'
self.assertFalse(volume_utils.hosts_are_equivalent(host_1, host_2)) self.assertFalse(volume_utils.hosts_are_equivalent(host_1, host_2))
@mock.patch('cinder.volume.utils.CONF')
def test_extract_id_from_volume_name_vol_id_pattern(self, conf_mock):
conf_mock.volume_name_template = 'volume-%s'
vol_id = 'd8cd1feb-2dcc-404d-9b15-b86fe3bec0a1'
vol_name = conf_mock.volume_name_template % vol_id
result = volume_utils.extract_id_from_volume_name(vol_name)
self.assertEqual(vol_id, result)
@mock.patch('cinder.volume.utils.CONF')
def test_extract_id_from_volume_name_vol_id_vol_pattern(self, conf_mock):
conf_mock.volume_name_template = 'volume-%s-volume'
vol_id = 'd8cd1feb-2dcc-404d-9b15-b86fe3bec0a1'
vol_name = conf_mock.volume_name_template % vol_id
result = volume_utils.extract_id_from_volume_name(vol_name)
self.assertEqual(vol_id, result)
@mock.patch('cinder.volume.utils.CONF')
def test_extract_id_from_volume_name_id_vol_pattern(self, conf_mock):
conf_mock.volume_name_template = '%s-volume'
vol_id = 'd8cd1feb-2dcc-404d-9b15-b86fe3bec0a1'
vol_name = conf_mock.volume_name_template % vol_id
result = volume_utils.extract_id_from_volume_name(vol_name)
self.assertEqual(vol_id, result)
@mock.patch('cinder.volume.utils.CONF')
def test_extract_id_from_volume_name_no_match(self, conf_mock):
conf_mock.volume_name_template = '%s-volume'
vol_name = 'd8cd1feb-2dcc-404d-9b15-b86fe3bec0a1'
result = volume_utils.extract_id_from_volume_name(vol_name)
self.assertIsNone(result)
vol_name = 'blahblahblah'
result = volume_utils.extract_id_from_volume_name(vol_name)
self.assertIsNone(result)
@mock.patch('cinder.db.sqlalchemy.api.resource_exists', return_value=True) @mock.patch('cinder.db.sqlalchemy.api.resource_exists', return_value=True)
def test_check_managed_volume_already_managed(self, exists_mock): def test_check_managed_volume_already_managed(self, exists_mock):
id_ = 'd8cd1feb-2dcc-404d-9b15-b86fe3bec0a1' id_ = 'd8cd1feb-2dcc-404d-9b15-b86fe3bec0a1'
vol_id = 'volume-' + id_ result = volume_utils.check_already_managed_volume(id_)
result = volume_utils.check_already_managed_volume(vol_id)
self.assertTrue(result)
exists_mock.assert_called_once_with(mock.ANY, models.Volume, id_)
@mock.patch('cinder.db.sqlalchemy.api.resource_exists', return_value=True)
def test_check_already_managed_with_vol_id_vol_pattern(self, exists_mock):
template = 'volume-%s-volume'
self.override_config('volume_name_template', template)
id_ = 'd8cd1feb-2dcc-404d-9b15-b86fe3bec0a1'
vol_id = template % id_
result = volume_utils.check_already_managed_volume(vol_id)
self.assertTrue(result)
exists_mock.assert_called_once_with(mock.ANY, models.Volume, id_)
@mock.patch('cinder.db.sqlalchemy.api.resource_exists', return_value=True)
def test_check_already_managed_with_id_vol_pattern(self, exists_mock):
template = '%s-volume'
self.override_config('volume_name_template', template)
id_ = 'd8cd1feb-2dcc-404d-9b15-b86fe3bec0a1'
vol_id = template % id_
result = volume_utils.check_already_managed_volume(vol_id)
self.assertTrue(result) self.assertTrue(result)
exists_mock.assert_called_once_with(mock.ANY, models.Volume, id_) exists_mock.assert_called_once_with(mock.ANY, models.Volume, id_)
@mock.patch('cinder.db.sqlalchemy.api.resource_exists', return_value=False) @mock.patch('cinder.db.sqlalchemy.api.resource_exists', return_value=False)
def test_check_managed_volume_not_managed_cinder_like_name(self, def test_check_managed_volume_not_managed_proper_uuid(self, exists_mock):
exists_mock):
id_ = 'd8cd1feb-2dcc-404d-9b15-b86fe3bec0a1' id_ = 'd8cd1feb-2dcc-404d-9b15-b86fe3bec0a1'
vol_id = 'volume-' + id_ result = volume_utils.check_already_managed_volume(id_)
result = volume_utils.check_already_managed_volume(vol_id)
self.assertFalse(result) self.assertFalse(result)
exists_mock.assert_called_once_with(mock.ANY, models.Volume, id_) exists_mock.assert_called_once_with(mock.ANY, models.Volume, id_)
def test_check_managed_volume_not_managed(self): def test_check_managed_volume_not_managed_invalid_id(self):
result = volume_utils.check_already_managed_volume('test-volume') result = volume_utils.check_already_managed_volume(1)
self.assertFalse(result)
result = volume_utils.check_already_managed_volume('not-a-uuid')
self.assertFalse(result) self.assertFalse(result)
def test_check_managed_volume_not_managed_id_like_uuid(self): @mock.patch('cinder.volume.utils.CONF')
result = volume_utils.check_already_managed_volume('volume-d8cd1fe') def test_extract_id_from_snapshot_name(self, conf_mock):
self.assertFalse(result) conf_mock.snapshot_name_template = '%s-snapshot'
snap_id = 'd8cd1feb-2dcc-404d-9b15-b86fe3bec0a1'
snap_name = conf_mock.snapshot_name_template % snap_id
result = volume_utils.extract_id_from_snapshot_name(snap_name)
self.assertEqual(snap_id, result)
@mock.patch('cinder.volume.utils.CONF')
def test_extract_id_from_snapshot_name_no_match(self, conf_mock):
conf_mock.snapshot_name_template = '%s-snapshot'
snap_name = 'd8cd1feb-2dcc-404d-9b15-b86fe3bec0a1'
result = volume_utils.extract_id_from_snapshot_name(snap_name)
self.assertIsNone(result)
snap_name = 'blahblahblah'
result = volume_utils.extract_id_from_snapshot_name(snap_name)
self.assertIsNone(result)
def test_paginate_entries_list_with_marker(self):
entries = [{'reference': {'name': 'vol03'}, 'size': 1},
{'reference': {'name': 'vol01'}, 'size': 3},
{'reference': {'name': 'vol02'}, 'size': 3},
{'reference': {'name': 'vol04'}, 'size': 2},
{'reference': {'name': 'vol06'}, 'size': 3},
{'reference': {'name': 'vol07'}, 'size': 1},
{'reference': {'name': 'vol05'}, 'size': 1}]
expected = [{'reference': {'name': 'vol04'}, 'size': 2},
{'reference': {'name': 'vol03'}, 'size': 1},
{'reference': {'name': 'vol05'}, 'size': 1}]
res = volume_utils.paginate_entries_list(entries, {'name': 'vol02'}, 3,
1, ['size', 'reference'],
['desc', 'asc'])
self.assertEqual(expected, res)
def test_paginate_entries_list_without_marker(self):
entries = [{'reference': {'name': 'vol03'}, 'size': 1},
{'reference': {'name': 'vol01'}, 'size': 3},
{'reference': {'name': 'vol02'}, 'size': 3},
{'reference': {'name': 'vol04'}, 'size': 2},
{'reference': {'name': 'vol06'}, 'size': 3},
{'reference': {'name': 'vol07'}, 'size': 1},
{'reference': {'name': 'vol05'}, 'size': 1}]
expected = [{'reference': {'name': 'vol07'}, 'size': 1},
{'reference': {'name': 'vol06'}, 'size': 3},
{'reference': {'name': 'vol05'}, 'size': 1}]
res = volume_utils.paginate_entries_list(entries, None, 3, None,
['reference'], ['desc'])
self.assertEqual(expected, res)
def test_paginate_entries_list_marker_not_found(self):
entries = [{'reference': {'name': 'vol03'}, 'size': 1},
{'reference': {'name': 'vol01'}, 'size': 3}]
self.assertRaises(exception.InvalidInput,
volume_utils.paginate_entries_list,
entries, {'name': 'vol02'}, 3, None,
['size', 'reference'], ['desc', 'asc'])
def test_convert_config_string_to_dict(self): def test_convert_config_string_to_dict(self):
test_string = "{'key-1'='val-1' 'key-2'='val-2' 'key-3'='val-3'}" test_string = "{'key-1'='val-1' 'key-2'='val-2' 'key-3'='val-3'}"

View File

@ -791,6 +791,84 @@ class LVMVolumeDriverTestCase(DriverTestCase):
ret = self.volume.driver.unmanage(volume) ret = self.volume.driver.unmanage(volume)
self.assertIsNone(ret) self.assertIsNone(ret)
def test_lvm_get_manageable_volumes(self):
cinder_vols = [{'id': '00000000-0000-0000-0000-000000000000'}]
lvs = [{'name': 'volume-00000000-0000-0000-0000-000000000000',
'size': '1.75'},
{'name': 'volume-00000000-0000-0000-0000-000000000001',
'size': '3.0'},
{'name': 'snapshot-00000000-0000-0000-0000-000000000002',
'size': '2.2'},
{'name': 'myvol', 'size': '4.0'}]
self.volume.driver.vg = mock.Mock()
self.volume.driver.vg.get_volumes.return_value = lvs
self.volume.driver.vg.lv_is_snapshot.side_effect = [False, False,
True, False]
self.volume.driver.vg.lv_is_open.side_effect = [True, False]
res = self.volume.driver.get_manageable_volumes(cinder_vols, None,
1000, 0,
['size'], ['asc'])
exp = [{'size': 2, 'reason_not_safe': None, 'extra_info': None,
'reference': {'source-name':
'volume-00000000-0000-0000-0000-000000000000'},
'cinder_id': '00000000-0000-0000-0000-000000000000',
'safe_to_manage': False, 'reason_not_safe': 'already managed'},
{'size': 3, 'reason_not_safe': 'volume in use',
'reference': {'source-name':
'volume-00000000-0000-0000-0000-000000000001'},
'safe_to_manage': False, 'cinder_id': None,
'extra_info': None},
{'size': 4, 'reason_not_safe': None,
'safe_to_manage': True, 'reference': {'source-name': 'myvol'},
'cinder_id': None, 'extra_info': None}]
self.assertEqual(exp, res)
def test_lvm_get_manageable_snapshots(self):
cinder_snaps = [{'id': '00000000-0000-0000-0000-000000000000'}]
lvs = [{'name': 'snapshot-00000000-0000-0000-0000-000000000000',
'size': '1.75'},
{'name': 'volume-00000000-0000-0000-0000-000000000001',
'size': '3.0'},
{'name': 'snapshot-00000000-0000-0000-0000-000000000002',
'size': '2.2'},
{'name': 'mysnap', 'size': '4.0'}]
self.volume.driver.vg = mock.Mock()
self.volume.driver.vg.get_volumes.return_value = lvs
self.volume.driver.vg.lv_is_snapshot.side_effect = [True, False, True,
True]
self.volume.driver.vg.lv_is_open.side_effect = [True, False]
self.volume.driver.vg.lv_get_origin.side_effect = [
'volume-00000000-0000-0000-0000-000000000000',
'volume-00000000-0000-0000-0000-000000000002',
'myvol']
res = self.volume.driver.get_manageable_snapshots(cinder_snaps, None,
1000, 0,
['size'], ['asc'])
exp = [{'size': 2, 'reason_not_safe': 'already managed',
'reference':
{'source-name':
'snapshot-00000000-0000-0000-0000-000000000000'},
'safe_to_manage': False, 'extra_info': None,
'cinder_id': '00000000-0000-0000-0000-000000000000',
'source_reference':
{'source-name':
'volume-00000000-0000-0000-0000-000000000000'}},
{'size': 3, 'reason_not_safe': 'snapshot in use',
'reference':
{'source-name':
'snapshot-00000000-0000-0000-0000-000000000002'},
'safe_to_manage': False, 'extra_info': None,
'cinder_id': None,
'source_reference':
{'source-name':
'volume-00000000-0000-0000-0000-000000000002'}},
{'size': 4, 'reason_not_safe': None,
'reference': {'source-name': 'mysnap'},
'safe_to_manage': True, 'cinder_id': None,
'source_reference': {'source-name': 'myvol'},
'extra_info': None}]
self.assertEqual(exp, res)
# Global setting, LVM setting, expected outcome # Global setting, LVM setting, expected outcome
@ddt.data((10.0, 2.0, 2.0)) @ddt.data((10.0, 2.0, 2.0))
@ddt.data((10.0, None, 10.0)) @ddt.data((10.0, None, 10.0))

View File

@ -178,6 +178,12 @@ class LVMVolumeDriver(driver.VolumeDriver):
return snapshot_name return snapshot_name
return '_' + snapshot_name return '_' + snapshot_name
def _unescape_snapshot(self, snapshot_name):
# Undo snapshot name change done by _escape_snapshot()
if not snapshot_name.startswith('_snapshot'):
return snapshot_name
return snapshot_name[1:]
def _create_volume(self, name, size, lvm_type, mirror_count, vg=None): def _create_volume(self, name, size, lvm_type, mirror_count, vg=None):
vg_ref = self.vg vg_ref = self.vg
if vg is not None: if vg is not None:
@ -586,7 +592,8 @@ class LVMVolumeDriver(driver.VolumeDriver):
lv_name = existing_ref['source-name'] lv_name = existing_ref['source-name']
self.vg.get_volume(lv_name) self.vg.get_volume(lv_name)
if volutils.check_already_managed_volume(lv_name): vol_id = volutils.extract_id_from_volume_name(lv_name)
if volutils.check_already_managed_volume(vol_id):
raise exception.ManageExistingAlreadyManaged(volume_ref=lv_name) raise exception.ManageExistingAlreadyManaged(volume_ref=lv_name)
# Attempt to rename the LV to match the OpenStack internal name. # Attempt to rename the LV to match the OpenStack internal name.
@ -654,6 +661,61 @@ class LVMVolumeDriver(driver.VolumeDriver):
existing_ref = {"source-name": existing_ref} existing_ref = {"source-name": existing_ref}
return self.manage_existing(snapshot_temp, existing_ref) return self.manage_existing(snapshot_temp, existing_ref)
def _get_manageable_resource_info(self, cinder_resources, resource_type,
marker, limit, offset, sort_keys,
sort_dirs):
entries = []
lvs = self.vg.get_volumes()
cinder_ids = [resource['id'] for resource in cinder_resources]
for lv in lvs:
is_snap = self.vg.lv_is_snapshot(lv['name'])
if ((resource_type == 'volume' and is_snap) or
(resource_type == 'snapshot' and not is_snap)):
continue
if resource_type == 'volume':
potential_id = volutils.extract_id_from_volume_name(lv['name'])
else:
unescape = self._unescape_snapshot(lv['name'])
potential_id = volutils.extract_id_from_snapshot_name(unescape)
lv_info = {'reference': {'source-name': lv['name']},
'size': int(math.ceil(float(lv['size']))),
'cinder_id': None,
'extra_info': None}
if potential_id in cinder_ids:
lv_info['safe_to_manage'] = False
lv_info['reason_not_safe'] = 'already managed'
lv_info['cinder_id'] = potential_id
elif self.vg.lv_is_open(lv['name']):
lv_info['safe_to_manage'] = False
lv_info['reason_not_safe'] = '%s in use' % resource_type
else:
lv_info['safe_to_manage'] = True
lv_info['reason_not_safe'] = None
if resource_type == 'snapshot':
origin = self.vg.lv_get_origin(lv['name'])
lv_info['source_reference'] = {'source-name': origin}
entries.append(lv_info)
return volutils.paginate_entries_list(entries, marker, limit, offset,
sort_keys, sort_dirs)
def get_manageable_volumes(self, cinder_volumes, marker, limit, offset,
sort_keys, sort_dirs):
return self._get_manageable_resource_info(cinder_volumes, 'volume',
marker, limit,
offset, sort_keys, sort_dirs)
def get_manageable_snapshots(self, cinder_snapshots, marker, limit, offset,
sort_keys, sort_dirs):
return self._get_manageable_resource_info(cinder_snapshots, 'snapshot',
marker, limit,
offset, sort_keys, sort_dirs)
def retype(self, context, volume, new_type, diff, host): def retype(self, context, volume, new_type, diff, host):
"""Retypes a volume, allow QoS and extra_specs change.""" """Retypes a volume, allow QoS and extra_specs change."""

View File

@ -16,7 +16,9 @@
import ast import ast
import functools
import math import math
import operator
import re import re
import time import time
import uuid import uuid
@ -664,28 +666,85 @@ def read_proc_mounts():
return mounts.readlines() return mounts.readlines()
def _extract_id(vol_name): def extract_id_from_volume_name(vol_name):
regex = re.compile( regex = re.compile(
CONF.volume_name_template.replace('%s', '(?P<uuid>.+)')) CONF.volume_name_template.replace('%s', '(?P<uuid>.+)'))
match = regex.match(vol_name) match = regex.match(vol_name)
return match.group('uuid') if match else None return match.group('uuid') if match else None
def check_already_managed_volume(vol_name): def check_already_managed_volume(vol_id):
"""Check cinder db for already managed volume. """Check cinder db for already managed volume.
:param vol_name: volume name parameter :param vol_id: volume id parameter
:returns: bool -- return True, if db entry with specified :returns: bool -- return True, if db entry with specified
volume name exist, otherwise return False volume id exists, otherwise return False
""" """
vol_id = _extract_id(vol_name)
try: try:
return (vol_id and uuid.UUID(vol_id, version=4) and return (vol_id and isinstance(vol_id, six.string_types) and
uuid.UUID(vol_id, version=4) and
objects.Volume.exists(context.get_admin_context(), vol_id)) objects.Volume.exists(context.get_admin_context(), vol_id))
except ValueError: except ValueError:
return False return False
def extract_id_from_snapshot_name(snap_name):
"""Return a snapshot's ID from its name on the backend."""
regex = re.compile(
CONF.snapshot_name_template.replace('%s', '(?P<uuid>.+)'))
match = regex.match(snap_name)
return match.group('uuid') if match else None
def paginate_entries_list(entries, marker, limit, offset, sort_keys,
sort_dirs):
"""Paginate a list of entries.
:param entries: list of dictionaries
:marker: The last element previously returned
:limit: The maximum number of items to return
:offset: The number of items to skip from the marker or from the first
element.
:sort_keys: A list of keys in the dictionaries to sort by
:sort_dirs: A list of sort directions, where each is either 'asc' or 'dec'
"""
comparers = [(operator.itemgetter(key.strip()), multiplier)
for (key, multiplier) in zip(sort_keys, sort_dirs)]
def comparer(left, right):
for fn, d in comparers:
left_val = fn(left)
right_val = fn(right)
if isinstance(left_val, dict):
left_val = sorted(left_val.values())[0]
if isinstance(right_val, dict):
right_val = sorted(right_val.values())[0]
if left_val == right_val:
continue
if d == 'asc':
return -1 if left_val < right_val else 1
else:
return -1 if left_val > right_val else 1
else:
return 0
sorted_entries = sorted(entries, key=functools.cmp_to_key(comparer))
start_index = 0
if offset is None:
offset = 0
if marker:
start_index = -1
for i, entry in enumerate(sorted_entries):
if entry['reference'] == marker:
start_index = i + 1
break
if start_index < 0:
msg = _('marker not found: %s') % marker
raise exception.InvalidInput(reason=msg)
range_end = start_index + limit
return sorted_entries[start_index + offset:range_end + offset]
def convert_config_string_to_dict(config_string): def convert_config_string_to_dict(config_string):
"""Convert config file replication string to a dict. """Convert config file replication string to a dict.