fuel-octane/octane/tests/test_util_node.py

258 lines
8.5 KiB
Python

# 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.
import contextlib
import io
import mock
import pytest
from octane.util import node as node_util
from octane.util import ssh
NODES = [
{'id': '1',
'fqdn': 'node-1.domain.tld',
'network_data': [{'name': 'management', 'ip': '10.20.0.2'},
{'name': 'public', 'ip': '172.167.0.2'}]},
{'id': '2',
'fqdn': 'node-2.domain.tld',
'network_data': [{'name': 'management', 'ip': '10.20.0.3'},
{'name': 'public', 'ip': '172.167.0.3'}]},
{'id': '3',
'fqdn': 'node-3.domain.tld',
'network_data': [{'name': 'management', 'ip': '10.20.0.4'},
{'name': 'public', 'ip': '172.167.0.4'}]},
]
@pytest.mark.parametrize('node_data,network_name,expected_ip', [
(NODES[0], 'management', '10.20.0.2'),
(NODES[0], 'storage', None),
({'network_data': []}, 'management', None),
])
def test_get_ip(node_data, network_name, expected_ip):
node = create_node(node_data)
ip = node_util.get_ip(network_name, node)
assert ip == expected_ip
def create_node(data):
return mock.Mock(data=data, spec_set=['data', 'env'])
@pytest.fixture
def nodes():
return map(create_node, NODES)
@pytest.mark.parametrize("network_name,expected_ips", [
('management', ['10.20.0.2', '10.20.0.3', '10.20.0.4']),
('public', ['172.167.0.2', '172.167.0.3', '172.167.0.4']),
])
def test_get_ips(nodes, network_name, expected_ips):
ips = node_util.get_ips(network_name, nodes)
assert ips == expected_ips
def test_get_hostnames(nodes):
hostnames = node_util.get_hostnames(nodes)
assert hostnames == ['node-1.domain.tld',
'node-2.domain.tld',
'node-3.domain.tld']
def test_tar_files(node, mock_ssh_popen, mock_open):
content = b'fake data\nin\nthe\narchive'
proc = mock_ssh_popen.return_value.__enter__.return_value
proc.stdout = io.BytesIO(content)
buf = io.BytesIO()
mock_open.return_value.write.side_effect = buf.write
node_util.tar_files('filename', node, 'a.file', 'b.file')
mock_ssh_popen.assert_called_once_with(
['tar', '-czvP', 'a.file', 'b.file'],
stdout=ssh.PIPE, node=node)
mock_open.assert_called_once_with('filename', 'wb')
assert buf.getvalue() == content
def test_untar_files(node, mock_ssh_popen, mock_open):
content = b'fake data\nin\nthe\narchive'
proc = mock_ssh_popen.return_value.__enter__.return_value
buf = io.BytesIO()
proc.stdin.write = buf.write
mock_open.return_value = io.BytesIO(content)
node_util.untar_files('filename', node)
mock_ssh_popen.assert_called_once_with(['tar', '-xzv', '-C', '/'],
stdin=ssh.PIPE, node=node)
mock_open.assert_called_once_with('filename', 'rb')
assert buf.getvalue() == content
@contextlib.contextmanager
def _check_upgrade_levels(mocker, node, content, expected_content):
mock_sftp = mocker.patch('octane.util.ssh.sftp')
old = io.BytesIO(content)
mock_new = mocker.Mock()
buf = io.BytesIO()
mock_new.write = buf.write
mock_update_file = mocker.patch('octane.util.ssh.update_file')
mock_update_file.return_value.__enter__.return_value = (old, mock_new)
yield
mock_sftp.assert_called_once_with(node)
mock_update_file.assert_called_once_with(mock_sftp.return_value,
'/etc/nova/nova.conf')
assert buf.getvalue() == expected_content
NOVA_DEFAULT = b"#\u0444\n[DEFAULT]\ndebug = True\n"
NOVA_WITH_EMPTY_LEVELS = NOVA_DEFAULT + b"[upgrade_levels]\n"
NOVA_WITH_KILO_LEVELS = NOVA_WITH_EMPTY_LEVELS + b"compute=kilo\n"
@pytest.mark.parametrize("content,expected_content", [
(NOVA_DEFAULT, NOVA_DEFAULT),
(NOVA_WITH_EMPTY_LEVELS, NOVA_WITH_KILO_LEVELS),
])
def test_add_compute_upgrade_levels(mocker, node, content, expected_content):
with _check_upgrade_levels(mocker, node, content, expected_content):
node_util.add_compute_upgrade_levels(node, 'kilo')
@pytest.mark.parametrize("content,expected_content", [
(NOVA_DEFAULT, NOVA_DEFAULT),
(NOVA_WITH_EMPTY_LEVELS, NOVA_WITH_EMPTY_LEVELS),
(NOVA_WITH_KILO_LEVELS, NOVA_WITH_EMPTY_LEVELS),
])
def test_remove_compute_upgrade_levels(mocker, node, content,
expected_content):
with _check_upgrade_levels(mocker, node, content, expected_content):
node_util.remove_compute_upgrade_levels(node)
NOVA_LIVE_MIGRATION_FLAG = b" live_migration_flag="
NOVA_LIVE_MIGRATION_ENABLED = NOVA_LIVE_MIGRATION_FLAG + \
b"FLAG1,VIR_MIGRATE_LIVE,FLAG2\n"
NOVA_NO_LIVE_MIGRATION_FLAG = b"no_live_migration_flag\n"
NOVA_EMPTY = b""
@pytest.mark.parametrize("content,expected_res", [
(NOVA_LIVE_MIGRATION_ENABLED, True),
(NOVA_LIVE_MIGRATION_FLAG, False),
(NOVA_NO_LIVE_MIGRATION_FLAG, False),
(NOVA_EMPTY, False),
])
def test_is_live_migration_supported(mocker, node, content, expected_res):
mock_sftp = mocker.patch("octane.util.ssh.sftp")
mock_sftp.return_value.open.return_value = io.BytesIO(content)
res = node_util.is_live_migration_supported(node)
assert res == expected_res
@pytest.mark.parametrize('node_data,fuel_version,expected_name', [
(NODES[0], '6.0', 'node-1'),
(NODES[0], '6.1', 'node-1.domain.tld'),
(NODES[0], 'invalid', None)
])
def test_get_nova_node_handle(mocker, node_data, fuel_version, expected_name):
node = create_node(node_data)
node.env.data.get.return_value = fuel_version
if expected_name:
name = node_util.get_nova_node_handle(node)
assert name == expected_name
else:
with pytest.raises(Exception):
node_util.get_nova_node_handle(node)
@pytest.mark.parametrize('stdout,nova_services_to_restart', [
(
" [ + ] nova-service-1\n"
" [ + ] nova-service-2\n"
" [ - ] nova-service-3\n"
" [ - ] not-nova-service-1\n"
" [ + ] not-nova-service-2\n"
" [ + ] nova-service-4\n",
[
'nova-service-1',
'nova-service-2',
'nova-service-4',
]
)
])
def test_restart_nova_services(mocker, node, stdout, nova_services_to_restart):
call_output_mock = mocker.patch(
"octane.util.ssh.call_output", return_value=stdout)
call_mock = mocker.patch("octane.util.ssh.call")
node_util.restart_nova_services(node)
for service in nova_services_to_restart:
call_mock.assert_any_call(["service", service, "restart"], node=node)
call_output_mock.assert_called_once_with(
["service", "--status-all"], node=node)
@pytest.mark.parametrize('online', [True, False])
@pytest.mark.parametrize('exception', [True, False])
def test_restart_mcollective_node(mocker, online, exception):
class TestException(Exception):
pass
side_effect = TestException('test exception')
log_patch = mocker.patch.object(node_util, 'LOG')
node = mock.Mock(id='node_test_id', data={'online': online})
ssh_call_mock = mocker.patch('octane.util.ssh.call')
if exception:
ssh_call_mock.side_effect = side_effect
node_util.restart_mcollective_on_node(node)
if online:
ssh_call_mock.assert_called_once_with(
["service", "mcollective", "restart"], node=node)
else:
assert not ssh_call_mock.called
if not online:
log_patch.warning.assert_called_once_with(
"Not possible to restart mcollective on the offline node {0}",
node.id)
elif exception:
log_patch.warning.assert_called_once_with(
"Failed to restart mcollective on the node %s: %s",
node.id, side_effect)
else:
assert not log_patch.warning.called
@pytest.mark.parametrize('nodes_count', [0, 1, 10])
def test_restart_mcollective(mocker, nodes_count):
nodes = []
calls = []
for _ in range(nodes_count):
node = mock.Mock()
calls.append(mock.call(node))
nodes.append(node)
mock_restart_mcol = mocker.patch(
'octane.util.node.restart_mcollective_on_node')
node_util.restart_mcollective(nodes)
assert calls == mock_restart_mcol.call_args_list