From fb52800873eafe33ce5ce79b2fccad5e8c91b6ba Mon Sep 17 00:00:00 2001 From: Alexander Graul Date: Thu, 23 Aug 2018 15:40:02 +0200 Subject: [PATCH] Add test for OSD migration (different slots) Signed-off-by: Alexander Graul --- tests/unit/runners/test_proposal.py | 260 ++++++++++++++++++++++++++++ 1 file changed, 260 insertions(+) create mode 100644 tests/unit/runners/test_proposal.py diff --git a/tests/unit/runners/test_proposal.py b/tests/unit/runners/test_proposal.py new file mode 100644 index 000000000..1ee90366d --- /dev/null +++ b/tests/unit/runners/test_proposal.py @@ -0,0 +1,260 @@ +from srv.modules.runners import proposal +import pytest +from mock import patch, mock_open, call +import sys +from collections import namedtuple +sys.path.insert(0, 'srv/modules/pillar') + + +@pytest.fixture +def minions(): + ToReplace = namedtuple('ToReplace', ['fullpath', 'filename']) + return { + 'minion1': { + 'minion': ToReplace('/srv/pillar/ceph/proposals/profile-default/stack/default/ceph/minions/data1.ceph.yml-replace', 'data1.ceph.yml-replace'), + 'basepath': "/srv/pillar/ceph/proposals/profile-default/stack/default/ceph/minions/data1.ceph.yml", + 'basename': "data1.ceph.yml", + 'name': "data1.ceph", + 'num_replace': 2, + 'disks': [ + # Noise here means nothing special, just literally 'noise' in the dictionary + {'Device File': '/dev/vdc', 'Noise': True, 'Device Files': '/dev/vdc'}, + {'Device File': '/dev/vdb', 'Noise': True, 'Device Files': '/dev/vdb'}, + {'Device File': '/dev/vdd', 'Noise': True, 'Device Files': '/dev/vdd'} + ], + 'osds': { + '/dev/vdb': {'format': 'bluestore', 'replace': False, 'old': 'vdb'}, + '/dev/vdd': {'format': 'bluestore', 'replace': True, 'old': 'vdd'}, + '/dev/vdc': {'format': 'bluestore', 'replace': True, 'old': 'vdc'} + }, + 'expected': { 'ceph': {'storage': {'osds': { + '/dev/vdb': {'format': 'bluestore', 'old': 'vdb'}, + '/dev/vdc': {'format': 'bluestore', 'old': 'vdc'}, + '/dev/vdd': {'format': 'bluestore', 'old': 'vdd'}}}}} + }, + 'minion2': { + 'minion': ToReplace('/srv/pillar/ceph/proposal/profile-alternative/stack/default/minions/data2.ceph.yml-replace', 'data2.ceph.yml-replace'), + 'basepath': "/srv/pillar/ceph/proposal/profile-alternative/stack/default/minions/data2.ceph.yml", + 'basename': "data2.ceph.yml", + 'name': "data2.ceph", + 'num_replace': 2, + 'disks': [ + {'Device File': '/dev/vdc', 'Noise': True, 'Device Files': '/dev/vdc, /dev/v_d_c'}, + {'Device File': '/dev/vdb', 'Noise': True, 'Device Files': '/dev/vdb, /dev/v_d_b'}, + {'Device File': '/dev/vde', 'Noise': True, 'Device Files': '/dev/vde, /dev/v_d_e'}, + {'Device File': '/dev/vdf', 'Noise': True, 'Device Files': '/dev/vdf, /dev/v_d_f'} + ], + 'osds': { + '/dev/v_d_b': {'format': 'bluestore', 'old': 'v_d_b'}, + '/dev/v_d_d': {'format': 'bluestore', 'replace': True, 'old': 'v_d_d'}, + '/dev/v_d_c': {'format': 'bluestore', 'replace': True, 'old': 'v_d_c'}, + '/dev/v_d_e': {'format': 'bluestore', 'old': 'v_d_e'}}, + 'expected': { 'ceph': {'storage': {'osds': { + '/dev/v_d_b': {'format': 'bluestore', 'old': 'v_d_b'}, + '/dev/v_d_c': {'format': 'bluestore', 'old': 'v_d_c'}, + '/dev/v_d_e': {'format': 'bluestore', 'old': 'v_d_e'}, + '/dev/v_d_f': {'format': 'bluestore', 'old': 'v_d_d'}}}}} + }, + 'minion3': { + 'minion': ToReplace('/srv/pillar/ceph/proposal/profile-default/stack/default/minions/data3.ceph.yml', 'data3.ceph.yml'), + 'basepath': "/srv/pillar/ceph/proposal/profile-default/stack/default/minions/data3.ceph.yml", + 'basename': "data3.ceph.yml", + 'name': "data3.ceph", + 'num_replace': 1, + 'disks': [ + {'Device File': '/dev/vdb', 'Noise': True, 'Device Files': '/dev/vdb'}, + {'Device File': '/dev/vde', 'Noise': True, 'Device Files': '/dev/vde'} + ], + 'osds': { + '/dev/vdc': {'format': 'bluestore', 'old': 'vdc', 'replace': True}, + '/dev/vde': {'format': 'bluestore', 'old': 'vde'}}, + 'expected': { 'ceph': {'storage': {'osds': { + '/dev/vdb': {'format': 'bluestore', 'old': 'vdc'}, + '/dev/vde': {'format': 'bluestore', 'old': 'vde'}}}}} + }, + 'minion4': { + 'minion': ToReplace('/srv/pillar/ceph/proposal/profile-default/stack/default/minions/we.use.a-very?weird-naming.scheme/node.yml-replace', 'node.yml-replace'), + 'basepath': "/srv/pillar/ceph/proposal/profile-default/stack/default/minions/we.use.a-very?weird-naming.scheme/node.yml", + 'basename': "node.yml", + 'name': "node", + 'num_replace': 0, + 'disks': [ + {'Device File': '/dev/vdb', 'Noise': True, 'Device Files': '/dev/vdb'}, + {'Device File': '/dev/vdc', 'Noise': True, 'Device Files': '/dev/vdc'}, + {'Device File': '/dev/vdd', 'Noise': True, 'Device Files': '/dev/vdd'}, + {'Device File': '/dev/vdf', 'Noise': True, 'Device Files': '/dev/vdf'} + ], + 'osds': { + '/dev/vdb': {'format': 'bluestore', 'old': 'vdb'}, + '/dev/vdc': {'format': 'bluestore', 'old': 'vdc'}, + '/dev/vdd': {'format': 'bluestore', 'old': 'vdd'}, + '/dev/vdf': {'format': 'bluestore', 'old': 'vdf'}}, + 'expected': { 'ceph': {'storage': {'osds': { + '/dev/vdb': {'format': 'bluestore', 'old': 'vdb'}, + '/dev/vdc': {'format': 'bluestore', 'old': 'vdc'}, + '/dev/vdd': {'format': 'bluestore', 'old': 'vdd'}, + '/dev/vdf': {'format': 'bluestore', 'old': 'vdf'}}}}} + }, + } + +class TestProposalRunner(object): + + @patch('srv.modules.runners.proposal.isfile', autospec=True) + @patch('os.listdir', autospec=True) + def test_find_minions_to_replace(self, mock_listdir, mock_isfile): + mock_listdir.return_value = ['data1.ceph.yml-replace'] + mock_isfile.return_value = True + directory = '/srv/pillar/ceph/proposals/profile-default' + result = proposal._find_minions_to_replace(directory) + + assert result[0].filename == 'data1.ceph.yml-replace' + + @patch('srv.modules.runners.proposal.isfile', autospec=True) + @patch('os.listdir', autospec=True) + def test_find_multiple_minions_to_replace(self, mock_listdir, mock_isfile): + mock_listdir.return_value = ['data1.ceph.yml-replace', 'data2.ceph.yml-replace'] + mock_isfile.return_value = True + directory = '/srv/pillar/ceph/proposals/profile-default' + result = proposal._find_minions_to_replace(directory) + + assert result[0].filename == 'data1.ceph.yml-replace' + assert result[1].filename == 'data2.ceph.yml-replace' + + @patch('srv.modules.runners.proposal.isfile', autospec=True) + @patch('os.listdir', autospec=True) + def test_find_minions_to_replace_no_result(self, mock_listdir, mock_isfile): + mock_listdir.return_value = ['.data1.ceph.yml-replace.swp'] + mock_isfile.return_value = True + directory = '/srv/pillar/ceph/proposals/profile-default' + result = proposal._find_minions_to_replace(directory) + + assert not result + + class NoInitReplaceDisk(proposal.ReplaceDiskOn): + """ Don't populate attributes via private methods automatically """ + + def __init__(self, minion): + self.minion = minion + + # range upper limit is amount of minions in fixture + 1 + @pytest.mark.parametrize("execution_number", range(1, 5)) + def test_proposal_basepath(self, execution_number, minions): + minion = minions['minion{}'.format(execution_number)] + RD = self.NoInitReplaceDisk(minion['minion']) + + assert RD._proposal_basepath() == minion['basepath'] + + @pytest.mark.parametrize("execution_number", range(1, 5)) + def test_proposal_basename(self, execution_number, minions): + minion = minions['minion{}'.format(execution_number)] + RD = self.NoInitReplaceDisk(minion['minion']) + + assert RD._proposal_basename() == minion['basename'] + + @pytest.mark.parametrize("execution_number", range(1, 5)) + def test_minion_name_from_file(self, execution_number, minions): + minion = minions['minion{}'.format(execution_number)] + RD = self.NoInitReplaceDisk(minion['minion']) + RD.proposal_basename = minion['basename'] + + assert RD._minion_name_from_file() == minion['name'] + + @patch('srv.modules.runners.proposal.open', new_callable=mock_open) + @patch('yaml.safe_load') + def test_load_proposal(self, mock_yaml, mock_file, minions): + minion = minions['minion1']['minion'] + proposal_dict = "{'fake_dict': True}" + mock_yaml.return_value = proposal_dict + RD = self.NoInitReplaceDisk(minion) + + assert RD._load_proposal() == proposal_dict + mock_yaml.assert_called_once() + + @patch('salt.client.LocalClient', autospec=True) + def test_query_node_disks(self, mock_client, minions): + minion = minions['minion1']['minion'] + local_client = mock_client.return_value + call1 = call('data1.ceph', 'mine.flush', tgt_type='compound') + call2 = call('data1.ceph', 'mine.update', tgt_type='compound') + call3 = call('data1.ceph', 'cephdisks.list', tgt_type='compound') + RD = self.NoInitReplaceDisk(minion) + RD.name = 'data1.ceph' + + RD._query_node_disks() + + assert local_client.cmd.call_count == 3 + assert local_client.cmd.call_args_list[0] == call1 + assert local_client.cmd.call_args_list[1] == call2 + assert local_client.cmd.call_args_list[2] == call3 + + + @pytest.mark.parametrize("execution_number", range(1, 5)) + @patch('salt.client.LocalClient', autospec=True) + def test_prepare_device_file(self, mock_client, execution_number, minions): + minion = minions['minion{}'.format(execution_number)] + local_client = mock_client.return_value + + RD = self.NoInitReplaceDisk(minion) + RD.disks = minion['disks'] + RD.old_proposal_disks = minion['osds'].keys() + + RD.device_files = RD._prepare_device_files() + + for disk in minion['disks']: + assert disk['Device Files'].split(',')[-1].strip() in RD.device_files + + + @pytest.mark.parametrize("execution_number", range(1, 5)) + def test_strip_replace_flages(self, execution_number, minions): + minion = minions['minion{}'.format(execution_number)] + RD = self.NoInitReplaceDisk(minion['minion']) + RD.proposal = {'ceph': {'storage': {'osds': minion['osds']}}} + RD.flagged_replace = [x for x in minion['osds'] if 'replace' in minion['osds'][x]] + + RD._strip_replace_flags() + + for conf in RD.proposal['ceph']['storage']['osds'].values(): + assert 'replace' not in conf + + @pytest.mark.parametrize("execution_number", range(1, 5)) + def test_swap_disks_in_proposal(self, execution_number, minions): + minion = minions['minion{}'.format(execution_number)] + RD = self.NoInitReplaceDisk(minion['minion']) + RD.proposal = {'ceph': {'storage': {'osds': minion['osds']}}} + RD.flagged_replace = sorted([x for x in minion['osds'] if 'replace' in minion['osds'][x]]) + # The device file with the most underscores is the last one in the 'Device Files' list here + # In the real code, the correct device file gets returned from a salt execution module + RD.unused_disks = sorted([x['Device Files'][-1] for x in minion['disks'] + # disks in new slots + if x['Device File'][-1] not in minion['osds'] + # disks that are exchanged in place + or 'replace' in minion['osds'][x['Device Files'][-1]] + and minion['osds'][x['Device Files'][-1]]['replace'] is True]) + keys_to_add = RD.unused_disks[:] + + RD._swap_disks_in_proposal() + + for i in range(minion['num_replace']): + assert keys_to_add[i] in RD.proposal['ceph']['storage']['osds'] + + + @pytest.mark.parametrize("execution_number", range(1, 5)) + @patch('salt.client.LocalClient') + @patch('srv.modules.runners.proposal.open', new_callable=mock_open) + @patch('yaml.safe_load') + def test_replace(self, mock_yaml, mock_file, mock_client, execution_number, minions): + minion = minions['minion{}'.format(execution_number)] + mock_yaml.return_value = {'ceph': {'storage': {'osds': minion['osds']}}} + + RD = proposal.ReplaceDiskOn(minion['minion']) + # salt client has different returns so instead of settings its + # return value we set the disks and device_files directly and re-run + # methods that depend on those + RD.disks = minion['disks'] + RD.device_files = sorted([x['Device Files'].split(',')[-1].strip() for x in minion['disks']]) + RD.unused_disks = RD._unused_disks() + RD.replace() + + mock_yaml.assert_called() + assert RD.proposal == minion['expected'] +