225 lines
9.2 KiB
Python
225 lines
9.2 KiB
Python
# vim: tabstop=4 shiftwidth=4 softtabstop=4
|
|
|
|
# Copyright 2013 Cloudbase Solutions SRL
|
|
# Copyright 2013 Pedro Navarro Perez
|
|
# All Rights Reserved.
|
|
#
|
|
# 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.
|
|
|
|
"""
|
|
Unit tests for Windows Hyper-V virtual switch neutron driver
|
|
"""
|
|
|
|
import mock
|
|
from oslo.config import cfg
|
|
|
|
from neutron.plugins.hyperv.agent import hyperv_neutron_agent
|
|
from neutron.plugins.hyperv.agent import utilsfactory
|
|
from neutron.tests import base
|
|
|
|
cfg.CONF.import_opt('enable_metrics_collection',
|
|
'neutron.plugins.hyperv.agent.hyperv_neutron_agent',
|
|
'AGENT')
|
|
|
|
|
|
class TestHyperVNeutronAgent(base.BaseTestCase):
|
|
|
|
_FAKE_PORT_ID = 'fake_port_id'
|
|
|
|
def setUp(self):
|
|
super(TestHyperVNeutronAgent, self).setUp()
|
|
# Avoid rpc initialization for unit tests
|
|
cfg.CONF.set_override('rpc_backend',
|
|
'neutron.openstack.common.rpc.impl_fake')
|
|
|
|
utilsfactory._get_windows_version = mock.MagicMock(
|
|
return_value='6.2.0')
|
|
|
|
class MockFixedIntervalLoopingCall(object):
|
|
def __init__(self, f):
|
|
self.f = f
|
|
|
|
def start(self, interval=0):
|
|
self.f()
|
|
|
|
mock.patch('neutron.openstack.common.loopingcall.'
|
|
'FixedIntervalLoopingCall',
|
|
new=MockFixedIntervalLoopingCall)
|
|
|
|
self.agent = hyperv_neutron_agent.HyperVNeutronAgent()
|
|
self.agent.plugin_rpc = mock.Mock()
|
|
self.agent.sec_groups_agent = mock.MagicMock()
|
|
self.agent.context = mock.Mock()
|
|
self.agent.agent_id = mock.Mock()
|
|
|
|
fake_agent_state = {
|
|
'binary': 'neutron-hyperv-agent',
|
|
'host': 'fake_host_name',
|
|
'topic': 'N/A',
|
|
'configurations': {'vswitch_mappings': ['*:MyVirtualSwitch']},
|
|
'agent_type': 'HyperV agent',
|
|
'start_flag': True}
|
|
self.agent_state = fake_agent_state
|
|
|
|
def test_port_bound_enable_metrics(self):
|
|
cfg.CONF.set_override('enable_metrics_collection', True, 'AGENT')
|
|
self._test_port_bound(True)
|
|
|
|
def test_port_bound_no_metrics(self):
|
|
cfg.CONF.set_override('enable_metrics_collection', False, 'AGENT')
|
|
self._test_port_bound(False)
|
|
|
|
def _test_port_bound(self, enable_metrics):
|
|
port = mock.MagicMock()
|
|
mock_enable_metrics = mock.MagicMock()
|
|
net_uuid = 'my-net-uuid'
|
|
|
|
with mock.patch.multiple(
|
|
self.agent._utils,
|
|
connect_vnic_to_vswitch=mock.MagicMock(),
|
|
set_vswitch_port_vlan_id=mock.MagicMock(),
|
|
enable_port_metrics_collection=mock_enable_metrics):
|
|
|
|
self.agent._port_bound(port, net_uuid, 'vlan', None, None)
|
|
|
|
self.assertEqual(enable_metrics, mock_enable_metrics.called)
|
|
|
|
def test_port_unbound(self):
|
|
map = {
|
|
'network_type': 'vlan',
|
|
'vswitch_name': 'fake-vswitch',
|
|
'ports': [],
|
|
'vlan_id': 1}
|
|
net_uuid = 'my-net-uuid'
|
|
network_vswitch_map = (net_uuid, map)
|
|
with mock.patch.object(self.agent,
|
|
'_get_network_vswitch_map_by_port_id',
|
|
return_value=network_vswitch_map):
|
|
with mock.patch.object(
|
|
self.agent._utils,
|
|
'disconnect_switch_port'):
|
|
self.agent._port_unbound(net_uuid)
|
|
|
|
def test_port_enable_control_metrics_ok(self):
|
|
cfg.CONF.set_override('enable_metrics_collection', True, 'AGENT')
|
|
self.agent._port_metric_retries[self._FAKE_PORT_ID] = (
|
|
cfg.CONF.AGENT.metrics_max_retries)
|
|
|
|
with mock.patch.multiple(self.agent._utils,
|
|
can_enable_control_metrics=mock.MagicMock(),
|
|
enable_control_metrics=mock.MagicMock()):
|
|
|
|
self.agent._utils.can_enable_control_metrics.return_value = True
|
|
self.agent._port_enable_control_metrics()
|
|
self.agent._utils.enable_control_metrics.assert_called_with(
|
|
self._FAKE_PORT_ID)
|
|
|
|
self.assertNotIn(self._FAKE_PORT_ID, self.agent._port_metric_retries)
|
|
|
|
def test_port_enable_control_metrics_maxed(self):
|
|
cfg.CONF.set_override('enable_metrics_collection', True, 'AGENT')
|
|
cfg.CONF.set_override('metrics_max_retries', 3, 'AGENT')
|
|
self.agent._port_metric_retries[self._FAKE_PORT_ID] = (
|
|
cfg.CONF.AGENT.metrics_max_retries)
|
|
|
|
with mock.patch.multiple(self.agent._utils,
|
|
can_enable_control_metrics=mock.MagicMock(),
|
|
enable_control_metrics=mock.MagicMock()):
|
|
|
|
self.agent._utils.can_enable_control_metrics.return_value = False
|
|
for i in range(cfg.CONF.AGENT.metrics_max_retries + 1):
|
|
self.assertIn(self._FAKE_PORT_ID,
|
|
self.agent._port_metric_retries)
|
|
self.agent._port_enable_control_metrics()
|
|
|
|
self.assertNotIn(self._FAKE_PORT_ID, self.agent._port_metric_retries)
|
|
|
|
def test_treat_devices_added_returns_true_for_missing_device(self):
|
|
attrs = {'get_device_details.side_effect': Exception()}
|
|
self.agent.plugin_rpc.configure_mock(**attrs)
|
|
self.assertTrue(self.agent._treat_devices_added([{}]))
|
|
|
|
def mock_treat_devices_added(self, details, func_name):
|
|
"""Mock treat devices added.
|
|
|
|
:param details: the details to return for the device
|
|
:param func_name: the function that should be called
|
|
:returns: whether the named function was called
|
|
"""
|
|
attrs = {'get_device_details.return_value': details}
|
|
self.agent.plugin_rpc.configure_mock(**attrs)
|
|
with mock.patch.object(self.agent, func_name) as func:
|
|
self.assertFalse(self.agent._treat_devices_added([{}]))
|
|
return func.called
|
|
|
|
def test_treat_devices_added_updates_known_port(self):
|
|
details = mock.MagicMock()
|
|
details.__contains__.side_effect = lambda x: True
|
|
with mock.patch.object(self.agent.plugin_rpc,
|
|
"update_device_up") as func:
|
|
self.assertTrue(self.mock_treat_devices_added(details,
|
|
'_treat_vif_port'))
|
|
self.assertTrue(func.called)
|
|
|
|
def test_treat_devices_added_missing_port_id(self):
|
|
details = mock.MagicMock()
|
|
details.__contains__.side_effect = lambda x: False
|
|
with mock.patch.object(self.agent.plugin_rpc,
|
|
"update_device_up") as func:
|
|
self.assertFalse(self.mock_treat_devices_added(details,
|
|
'_treat_vif_port'))
|
|
self.assertFalse(func.called)
|
|
|
|
def test_treat_devices_removed_returns_true_for_missing_device(self):
|
|
attrs = {'update_device_down.side_effect': Exception()}
|
|
self.agent.plugin_rpc.configure_mock(**attrs)
|
|
self.assertTrue(self.agent._treat_devices_removed([{}]))
|
|
|
|
def mock_treat_devices_removed(self, port_exists):
|
|
details = dict(exists=port_exists)
|
|
attrs = {'update_device_down.return_value': details}
|
|
self.agent.plugin_rpc.configure_mock(**attrs)
|
|
with mock.patch.object(self.agent, '_port_unbound') as func:
|
|
self.assertFalse(self.agent._treat_devices_removed([{}]))
|
|
self.assertEqual(func.called, not port_exists)
|
|
|
|
def test_treat_devices_removed_unbinds_port(self):
|
|
self.mock_treat_devices_removed(False)
|
|
|
|
def test_treat_devices_removed_ignores_missing_port(self):
|
|
self.mock_treat_devices_removed(False)
|
|
|
|
def test_report_state(self):
|
|
with mock.patch.object(self.agent.state_rpc,
|
|
"report_state") as report_st:
|
|
self.agent._report_state()
|
|
report_st.assert_called_with(self.agent.context,
|
|
self.agent.agent_state)
|
|
self.assertNotIn("start_flag", self.agent.agent_state)
|
|
|
|
def test_main(self):
|
|
with mock.patch.object(hyperv_neutron_agent,
|
|
'HyperVNeutronAgent') as plugin:
|
|
with mock.patch.object(hyperv_neutron_agent.cfg, 'CONF') as cfg:
|
|
with mock.patch('eventlet.monkey_patch') as eventlet:
|
|
with mock.patch.object(
|
|
hyperv_neutron_agent,
|
|
'logging_config') as logging_config:
|
|
|
|
hyperv_neutron_agent.main()
|
|
|
|
self.assertTrue(cfg.called)
|
|
self.assertTrue(eventlet.called)
|
|
self.assertTrue(logging_config.setup_logging.called)
|
|
plugin.assert_has_calls([mock.call().daemon_loop()])
|