############################################################################## # Copyright (c) 2015-2017 Huawei Technologies Co.,Ltd and others. # # All rights reserved. This program and the accompanying materials # are made available under the terms of the Apache License, Version 2.0 # which accompanies this distribution, and is available at # http://www.apache.org/licenses/LICENSE-2.0 ############################################################################## import os import unittest import errno import mock from yardstick.common import constants as consts from yardstick.benchmark.contexts import node class NodeContextTestCase(unittest.TestCase): PREFIX = 'yardstick.benchmark.contexts.node' NODES_SAMPLE = "nodes_sample.yaml" NODES_DUPLICATE_SAMPLE = "nodes_duplicate_sample.yaml" def setUp(self): self.test_context = node.NodeContext() self.addCleanup(self._remove_contexts) self.os_path_join = os.path.join self.attrs = { 'name': 'foo', 'task_id': '1234567890', 'file': self._get_file_abspath(self.NODES_SAMPLE) } def _remove_contexts(self): if self.test_context in self.test_context.list: self.test_context._delete_context() def _get_file_abspath(self, filename): curr_path = os.path.dirname(os.path.abspath(__file__)) file_path = self.os_path_join(curr_path, filename) return file_path def test___init__(self): self.assertIsNone(self.test_context._name) self.assertIsNone(self.test_context.file_path) self.assertEqual(self.test_context.nodes, []) self.assertEqual(self.test_context.controllers, []) self.assertEqual(self.test_context.computes, []) self.assertEqual(self.test_context.baremetals, []) self.assertEqual(self.test_context.env, {}) self.assertEqual(self.test_context.attrs, {}) @mock.patch('{}.os.path.join'.format(PREFIX)) def test_init_negative(self, mock_path_join): special_path = '/foo/bar/error_file' error_path = self._get_file_abspath("error_file") def path_join(*args): if args == (consts.YARDSTICK_ROOT_PATH, error_path): return special_path return self.os_path_join(*args) # we can't count mock_path_join calls because # it can catch join calls for .pyc files. mock_path_join.side_effect = path_join self.test_context.read_config_file = read_mock = mock.Mock() read_calls = 0 with self.assertRaises(KeyError): self.test_context.init({}) self.assertEqual(read_mock.call_count, read_calls) attrs = { 'name': 'foo', 'task_id': '1234567890', 'file': error_path, } read_mock.side_effect = IOError(errno.EBUSY, 'busy') with self.assertRaises(IOError) as raised: self.test_context.init(attrs) read_calls += 1 self.assertEqual(read_mock.called, read_calls) self.assertIn(attrs['file'], self.test_context.file_path) self.assertEqual(raised.exception.errno, errno.EBUSY) self.assertEqual(str(raised.exception), str(read_mock.side_effect)) read_mock.side_effect = IOError(errno.ENOENT, 'not found') with self.assertRaises(IOError) as raised: self.test_context.init(attrs) read_calls += 2 self.assertEqual(read_mock.call_count, read_calls) self.assertEqual(self.test_context.file_path, special_path) self.assertEqual(raised.exception.errno, errno.ENOENT) self.assertEqual(str(raised.exception), str(read_mock.side_effect)) def test_read_config_file(self): self.test_context.init(self.attrs) self.assertIsNotNone(self.test_context.read_config_file()) def test__dispatch_script(self): self.test_context.init(self.attrs) self.test_context.env = {'bash': [{'script': 'dummy'}]} self.test_context._execute_script = mock.Mock() self.assertEqual(self.test_context._dispatch_script('bash'), None) def test__dispatch_ansible(self): self.test_context.init(self.attrs) self.test_context.env = {'ansible': [{'script': 'dummy'}]} self.test_context._do_ansible_job = mock.Mock() self.assertEqual(self.test_context._dispatch_ansible('ansible'), None) self.test_context.env = {} self.assertEqual(self.test_context._dispatch_ansible('ansible'), None) @mock.patch("{}.AnsibleCommon".format(PREFIX)) def test__do_ansible_job(self, *args): self.assertIsNone(self.test_context._do_ansible_job('dummy')) def test_init(self): self.test_context.init(self.attrs) self.assertEqual(self.test_context.name, "foo-12345678") self.assertEqual(len(self.test_context.nodes), 4) self.assertEqual(len(self.test_context.controllers), 2) self.assertEqual(len(self.test_context.computes), 1) self.assertEqual(self.test_context.computes[0]["name"], "node3") self.assertEqual(len(self.test_context.baremetals), 1) self.assertEqual(self.test_context.baremetals[0]["name"], "node4") def test__get_server_with_dict_attr_name(self): self.test_context.init(self.attrs) result = self.test_context._get_server({'name': 'node1.foo-12345678'}) self.assertIsNone(result, None) def test__get_server_not_found(self): self.test_context.init(self.attrs) self.assertIsNone(self.test_context._get_server('bar.foo-12345678')) def test__get_server_mismatch(self): self.test_context.init(self.attrs) self.assertIsNone(self.test_context._get_server('bar.foo1')) def test__get_server_duplicate(self): self.attrs['file'] = self._get_file_abspath( self.NODES_DUPLICATE_SAMPLE) self.test_context.init(self.attrs) with self.assertRaises(ValueError): self.test_context._get_server('node1.foo-12345678') def test__get_server_found(self): self.test_context.init(self.attrs) result = self.test_context._get_server('node1.foo-12345678') self.assertEqual(result['ip'], '10.229.47.137') self.assertEqual(result['name'], 'node1.foo-12345678') self.assertEqual(result['user'], 'root') self.assertEqual(result['key_filename'], '/root/.yardstick_key') @mock.patch('{}.NodeContext._dispatch_script'.format(PREFIX)) def test_deploy(self, dispatch_script_mock): obj = node.NodeContext() self.addCleanup(obj._delete_context) obj.env = { 'type': 'script' } obj.deploy() dispatch_script_mock.assert_called_once() @mock.patch('{}.NodeContext._dispatch_ansible'.format(PREFIX)) def test_deploy_anisible(self, dispatch_ansible_mock): obj = node.NodeContext() self.addCleanup(obj._delete_context) obj.env = { 'type': 'ansible' } obj.deploy() dispatch_ansible_mock.assert_called_once() @mock.patch('{}.NodeContext._dispatch_script'.format(PREFIX)) def test_undeploy(self, dispatch_script_mock): obj = node.NodeContext() obj.env = { 'type': 'script' } obj.undeploy() dispatch_script_mock.assert_called_once() @mock.patch('{}.NodeContext._dispatch_ansible'.format(PREFIX)) def test_undeploy_anisble(self, dispatch_ansible_mock): obj = node.NodeContext() obj.env = { 'type': 'ansible' } obj.undeploy() dispatch_ansible_mock.assert_called_once() @mock.patch('{}.ssh.SSH._put_file_shell'.format(PREFIX)) @mock.patch('{}.ssh.SSH.execute'.format(PREFIX)) def test_execute_remote_script(self, execute_mock, put_file_mock): obj = node.NodeContext() self.addCleanup(obj._delete_context) obj.env = {'prefix': 'yardstick.benchmark.scenarios.compute'} node_name_args = 'node5' obj.nodes = [{ 'name': node_name_args, 'user': 'ubuntu', 'ip': '10.10.10.10', 'pwd': 'ubuntu', }] info = {'script': 'computecapacity.bash'} execute_mock.return_value = (0, '', '') obj._execute_remote_script('node5', info) put_file_mock.assert_called_once() execute_mock.assert_called() @mock.patch('{}.NodeContext._execute_local_script'.format(PREFIX)) def test_execute_script_local(self, local_execute_mock): node_name = 'local' info = {} obj = node.NodeContext() self.addCleanup(obj._delete_context) obj._execute_script(node_name, info) local_execute_mock.assert_called_once() @mock.patch('{}.NodeContext._execute_remote_script'.format(PREFIX)) def test_execute_script_remote(self, remote_execute_mock): node_name = 'node5' info = {} obj = node.NodeContext() self.addCleanup(obj._delete_context) obj._execute_script(node_name, info) remote_execute_mock.assert_called_once() def test_get_script(self): script_args = 'hello.bash' info_args = { 'script': script_args } obj = node.NodeContext() self.addCleanup(obj._delete_context) script, options = obj._get_script(info_args) self.assertEqual(script_args, script) self.assertEqual('', options) def test_node_info(self): node_name_args = 'node5' obj = node.NodeContext() self.addCleanup(obj._delete_context) obj.nodes = [{'name': node_name_args, 'check': node_name_args}] node_info = obj._get_node_info(node_name_args) self.assertEqual(node_info.get('check'), node_name_args) @mock.patch('{}.ssh.SSH.wait'.format(PREFIX)) def test_get_client(self, wait_mock): node_name_args = 'node5' obj = node.NodeContext() self.addCleanup(obj._delete_context) obj.nodes = [{ 'name': node_name_args, 'user': 'ubuntu', 'ip': '10.10.10.10', 'pwd': 'ubuntu', }] obj._get_client(node_name_args) wait_mock.assert_called_once() def test_get_server(self): self.test_context.init(self.attrs) self.test_context._name = 'foo' self.test_context._task_id = '1234567890' self.test_context._name_task_id = '{}-{}'.format( self.test_context._name, self.test_context._task_id[:8]) self.assertEqual('foo-12345678', self.test_context.name) self.assertIsNotNone(self.test_context._task_id) result = self.test_context.get_server('node1.foo-12345678') self.assertEqual(result['ip'], '10.229.47.137') self.assertEqual(result['name'], 'node1.foo-12345678') self.assertEqual(result['user'], 'root') self.assertEqual(result['key_filename'], '/root/.yardstick_key') def test_get_server_server_not_in_context(self): self.test_context.init(self.attrs) with self.assertRaises(ValueError): self.test_context.get_server('my2.foo-12345678') def test_get_context_from_server(self): self.test_context._name = 'vnf1' self.test_context._task_id = '1234567890' self.test_context._name_task_id = '{}-{}'.format( self.test_context._name, self.test_context._task_id[:8]) self.test_context.nodes = [{'name': 'my', 'value': 100}] self.test_context.attrs = {'attr1': 200} self.assertIs( self.test_context.get_context_from_server('my.vnf1-12345678'), self.test_context) # TODO: Split this into more granular tests def test__get_network(self): network1 = { 'name': 'net_1', 'vld_id': 'vld111', 'segmentation_id': 'seg54', 'network_type': 'type_a', 'physical_network': 'phys', } network2 = { 'name': 'net_2', 'vld_id': 'vld999', } self.test_context.networks = { 'a': network1, 'b': network2, } attr_name = {} self.assertIsNone(self.test_context._get_network(attr_name)) attr_name = {'vld_id': 'vld777'} self.assertIsNone(self.test_context._get_network(attr_name)) self.assertIsNone(self.test_context._get_network(None)) attr_name = 'vld777' self.assertIsNone(self.test_context._get_network(attr_name)) attr_name = {'vld_id': 'vld999'} expected = { "name": 'net_2', "vld_id": 'vld999', "segmentation_id": None, "network_type": None, "physical_network": None, } result = self.test_context._get_network(attr_name) self.assertDictEqual(result, expected) attr_name = 'a' expected = network1 result = self.test_context._get_network(attr_name) self.assertDictEqual(result, expected)