Unit tests
This commit is contained in:
parent
6a087ad8c6
commit
ae26c6f642
@ -124,7 +124,10 @@ class TransportURLAdapter(os_adapters.RabbitMQRelationAdapter):
|
||||
|
||||
:returns: string transport URL
|
||||
"""
|
||||
hosts = self.hosts or [self.host]
|
||||
if self.hosts:
|
||||
hosts = self.hosts.split(',')
|
||||
else:
|
||||
hosts = [self.host]
|
||||
if hosts:
|
||||
transport_url_hosts = ','.join([
|
||||
"{}:{}@{}:{}".format(self.username,
|
||||
|
22
unit_tests/__init__.py
Normal file
22
unit_tests/__init__.py
Normal file
@ -0,0 +1,22 @@
|
||||
# Copyright 2018 Canonical Ltd
|
||||
#
|
||||
# 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 sys
|
||||
|
||||
sys.path.append('src')
|
||||
sys.path.append('src/lib')
|
||||
|
||||
# Mock out charmhelpers so that we can test without it.
|
||||
import charms_openstack.test_mocks # noqa
|
||||
charms_openstack.test_mocks.mock_charmhelpers()
|
151
unit_tests/test_dragent_handlers.py
Normal file
151
unit_tests/test_dragent_handlers.py
Normal file
@ -0,0 +1,151 @@
|
||||
# Copyright 2018 Canonical Ltd
|
||||
#
|
||||
# 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.
|
||||
|
||||
from __future__ import absolute_import
|
||||
from __future__ import print_function
|
||||
|
||||
import unittest
|
||||
|
||||
import mock
|
||||
|
||||
import reactive.dragent_handlers as handlers
|
||||
|
||||
|
||||
_when_args = {}
|
||||
_when_not_args = {}
|
||||
|
||||
|
||||
def mock_hook_factory(d):
|
||||
|
||||
def mock_hook(*args, **kwargs):
|
||||
|
||||
def inner(f):
|
||||
# remember what we were passed. Note that we can't actually
|
||||
# determine the class we're attached to, as the decorator only gets
|
||||
# the function.
|
||||
try:
|
||||
d[f.__name__].append(dict(args=args, kwargs=kwargs))
|
||||
except KeyError:
|
||||
d[f.__name__] = [dict(args=args, kwargs=kwargs)]
|
||||
return f
|
||||
return inner
|
||||
return mock_hook
|
||||
|
||||
|
||||
class TestDRAgentHandlers(unittest.TestCase):
|
||||
|
||||
@classmethod
|
||||
def setUpClass(cls):
|
||||
cls._patched_when = mock.patch('charms.reactive.when',
|
||||
mock_hook_factory(_when_args))
|
||||
cls._patched_when_started = cls._patched_when.start()
|
||||
cls._patched_when_not = mock.patch('charms.reactive.when_not',
|
||||
mock_hook_factory(_when_not_args))
|
||||
cls._patched_when_not_started = cls._patched_when_not.start()
|
||||
# force requires to rerun the mock_hook decorator:
|
||||
# try except is Python2/Python3 compatibility as Python3 has moved
|
||||
# reload to importlib.
|
||||
try:
|
||||
reload(handlers)
|
||||
except NameError:
|
||||
import importlib
|
||||
importlib.reload(handlers)
|
||||
|
||||
@classmethod
|
||||
def tearDownClass(cls):
|
||||
cls._patched_when.stop()
|
||||
cls._patched_when_started = None
|
||||
cls._patched_when = None
|
||||
cls._patched_when_not.stop()
|
||||
cls._patched_when_not_started = None
|
||||
cls._patched_when_not = None
|
||||
# and fix any breakage we did to the module
|
||||
try:
|
||||
reload(handlers)
|
||||
except NameError:
|
||||
import importlib
|
||||
importlib.reload(handlers)
|
||||
|
||||
def setUp(self):
|
||||
self._patches = {}
|
||||
self._patches_start = {}
|
||||
|
||||
def tearDown(self):
|
||||
for k, v in self._patches.items():
|
||||
v.stop()
|
||||
setattr(self, k, None)
|
||||
self._patches = None
|
||||
self._patches_start = None
|
||||
|
||||
def patch(self, obj, attr, return_value=None, side_effect=None):
|
||||
mocked = mock.patch.object(obj, attr)
|
||||
self._patches[attr] = mocked
|
||||
started = mocked.start()
|
||||
started.return_value = return_value
|
||||
started.side_effect = side_effect
|
||||
self._patches_start[attr] = started
|
||||
setattr(self, attr, started)
|
||||
|
||||
def test_registered_hooks(self):
|
||||
# test that the hooks actually registered the relation expressions that
|
||||
# are meaningful for this interface: this is to handle regressions.
|
||||
# The keys are the function names that the hook attaches to.
|
||||
when_patterns = {
|
||||
'publish_bgp_info': ('endpoint.bgp-speaker.changed',),
|
||||
'setup_amqp_req': ('amqp.connected', ),
|
||||
'render_configs': ('amqp.available', ),
|
||||
}
|
||||
when_not_patterns = {}
|
||||
# check the when hooks are attached to the expected functions
|
||||
for t, p in [(_when_args, when_patterns),
|
||||
(_when_not_args, when_not_patterns)]:
|
||||
for f, args in t.items():
|
||||
# check that function is in patterns
|
||||
self.assertTrue(f in p.keys(),
|
||||
"{} not found".format(f))
|
||||
# check that the lists are equal
|
||||
li = []
|
||||
for a in args:
|
||||
li += a['args'][:]
|
||||
self.assertEqual(sorted(li), sorted(p[f]),
|
||||
"{}: incorrect state registration".format(f))
|
||||
|
||||
def test_publish_bgp_info(self):
|
||||
_asn = 12345
|
||||
_bindings = ['bgp-speaker']
|
||||
self.patch(handlers.dragent, 'assess_status')
|
||||
self.patch(handlers.hookenv, 'config')
|
||||
self.config.return_value = _asn
|
||||
bgp = mock.MagicMock()
|
||||
handlers.publish_bgp_info(bgp)
|
||||
bgp.publish_info.assert_called_once_with(asn=_asn,
|
||||
passive=True,
|
||||
bindings=_bindings)
|
||||
|
||||
def test_setup_amqp_req(self):
|
||||
self.patch(handlers.dragent, 'assess_status')
|
||||
amqp = mock.MagicMock()
|
||||
handlers.setup_amqp_req(amqp)
|
||||
amqp.request_access.assert_called_once_with(
|
||||
username='dragent', vhost='openstack')
|
||||
|
||||
def test_render_configs(self):
|
||||
self.patch(handlers.dragent, 'render_configs')
|
||||
self.patch(handlers.dragent, 'assess_status')
|
||||
self.patch(handlers.dragent, 'upgrade_if_available')
|
||||
amqp = mock.MagicMock()
|
||||
handlers.render_configs(amqp)
|
||||
self.upgrade_if_available.assert_called_once_with((amqp,))
|
||||
self.render_configs.assert_called_once_with((amqp,))
|
||||
self.assess_status.assert_called_once()
|
110
unit_tests/test_lib_charm_openstack_dragent.py
Normal file
110
unit_tests/test_lib_charm_openstack_dragent.py
Normal file
@ -0,0 +1,110 @@
|
||||
# Copyright 2018 Canonical Ltd
|
||||
#
|
||||
# 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.
|
||||
|
||||
from __future__ import absolute_import
|
||||
from __future__ import print_function
|
||||
|
||||
import mock
|
||||
|
||||
import charm.openstack.dragent as dragent
|
||||
|
||||
import charms_openstack.test_utils as test_utils
|
||||
|
||||
|
||||
class Helper(test_utils.PatchHelper):
|
||||
|
||||
def setUp(self):
|
||||
super().setUp()
|
||||
self.patch_release(dragent.DRAgentCharm.release)
|
||||
self.SPEAKER_BINDING = "bgp-speaker"
|
||||
self.PROVIDER_BINDING = "provider"
|
||||
|
||||
|
||||
class TestOpenStackDRAgent(Helper):
|
||||
|
||||
def test_render_configs(self):
|
||||
self.patch_object(dragent.DRAgentCharm.singleton,
|
||||
"render_with_interfaces")
|
||||
dragent.render_configs("interfaces-list")
|
||||
self.render_with_interfaces.assert_called_once_with(
|
||||
"interfaces-list")
|
||||
|
||||
def test_bgp_speaker_bindings(self):
|
||||
self.assertEqual(dragent.bgp_speaker_bindings(),
|
||||
[self.SPEAKER_BINDING])
|
||||
|
||||
def test_speaker_ip(self):
|
||||
_ip = "10.0.0.10"
|
||||
self.patch_object(dragent.ch_ip, "get_relation_ip")
|
||||
self.get_relation_ip.return_value = _ip
|
||||
dra = dragent.DRAgentCharm()
|
||||
self.assertEqual(dragent.speaker_ip(dra), _ip)
|
||||
self.get_relation_ip.assert_called_once_with(self.SPEAKER_BINDING)
|
||||
|
||||
def test_provider_ip(self):
|
||||
_ip = "10.200.0.25"
|
||||
self.patch_object(dragent.ch_ip, "get_relation_ip")
|
||||
self.get_relation_ip.return_value = _ip
|
||||
dra = dragent.DRAgentCharm()
|
||||
self.assertEqual(dragent.provider_ip(dra), _ip)
|
||||
self.get_relation_ip.assert_called_once_with(self.PROVIDER_BINDING)
|
||||
|
||||
|
||||
class TestTransportURLAdapter(Helper):
|
||||
|
||||
def test_transport_url(self):
|
||||
_expected = "rabbit://user:pass@10.0.0.50:5672/vhost"
|
||||
amqp = mock.MagicMock()
|
||||
amqp.relation_name = "amqp"
|
||||
amqp.username.return_value = "user"
|
||||
amqp.vhost.return_value = "vhost"
|
||||
tua = dragent.TransportURLAdapter(amqp)
|
||||
tua.vip = None
|
||||
tua.password = "pass"
|
||||
tua.ssl_port = None
|
||||
|
||||
# Single
|
||||
tua.private_address = "10.0.0.50"
|
||||
self.assertEqual(tua.transport_url, _expected)
|
||||
|
||||
# Multiple
|
||||
_expected = ("rabbit://user:pass@10.200.0.20:5672,"
|
||||
"user:pass@10.200.0.30:5672/vhost")
|
||||
amqp.rabbitmq_hosts.return_value = ["10.200.0.20", "10.200.0.30"]
|
||||
self.assertEqual(tua.transport_url, _expected)
|
||||
|
||||
def test_port(self):
|
||||
_ssl_port = 2765
|
||||
_port = 5672
|
||||
amqp = mock.MagicMock()
|
||||
tua = dragent.TransportURLAdapter(amqp)
|
||||
# Default Port
|
||||
tua.ssl_port = None
|
||||
self.assertEqual(tua.port, _port)
|
||||
# SSL port
|
||||
tua.ssl_port = _ssl_port
|
||||
self.assertEqual(tua.port, _ssl_port)
|
||||
|
||||
|
||||
class TestDRAgentCharm(Helper):
|
||||
|
||||
def test_install(self):
|
||||
dra = dragent.DRAgentCharm()
|
||||
self.patch_object(dragent.charms_openstack.charm.OpenStackCharm,
|
||||
"configure_source")
|
||||
self.patch_object(dragent.charms_openstack.charm.OpenStackCharm,
|
||||
"install")
|
||||
dra.install()
|
||||
self.configure_source.assert_called_once_with()
|
||||
self.install.assert_called_once_with()
|
Loading…
x
Reference in New Issue
Block a user