Merge "Modernized backend tests"
This commit is contained in:
commit
4e49b5581f
@ -1,138 +0,0 @@
|
|||||||
# Copyright 2015 Infoblox Inc.
|
|
||||||
# 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.
|
|
||||||
import six
|
|
||||||
from mock import MagicMock
|
|
||||||
|
|
||||||
from designate import objects
|
|
||||||
from designate.tests.test_backend import BackendTestCase
|
|
||||||
from designate.backend.impl_infoblox import InfobloxBackend
|
|
||||||
from designate.exceptions import ConfigurationError
|
|
||||||
from designate.backend.impl_infoblox import ibexceptions
|
|
||||||
|
|
||||||
|
|
||||||
class InfobloxBackendTestCase(BackendTestCase):
|
|
||||||
|
|
||||||
def get_zone_fixture(self):
|
|
||||||
return super(InfobloxBackendTestCase, self).get_zone_fixture(
|
|
||||||
values={
|
|
||||||
'name': 'test.example.com.'
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
super(InfobloxBackendTestCase, self).setUp()
|
|
||||||
|
|
||||||
self.config(group='backend:infoblox',
|
|
||||||
wapi_url=None,
|
|
||||||
username=None,
|
|
||||||
password=None,
|
|
||||||
ns_group=None)
|
|
||||||
|
|
||||||
def get_target_fixture(self, masters=None, options=None):
|
|
||||||
if not masters:
|
|
||||||
masters = [{'host': '1.1.1.1', 'port': 53}]
|
|
||||||
|
|
||||||
if not options:
|
|
||||||
options = [{'key': 'wapi_url', 'value': 'test'},
|
|
||||||
{'key': 'username', 'value': 'test'},
|
|
||||||
{'key': 'password', 'value': 'test'},
|
|
||||||
{'key': 'ns_group', 'value': 'test'}]
|
|
||||||
|
|
||||||
return objects.PoolTarget.from_dict({
|
|
||||||
'id': '4588652b-50e7-46b9-b688-a9bad40a873e',
|
|
||||||
'type': 'infoblox',
|
|
||||||
'masters': masters,
|
|
||||||
'options': options
|
|
||||||
})
|
|
||||||
|
|
||||||
def set_up_backend(self, target=None):
|
|
||||||
if not target:
|
|
||||||
target = self.get_target_fixture()
|
|
||||||
|
|
||||||
self.backend = InfobloxBackend(target)
|
|
||||||
self.backend.start()
|
|
||||||
self.backend.infoblox = MagicMock()
|
|
||||||
|
|
||||||
def test_create_zone(self):
|
|
||||||
self.set_up_backend()
|
|
||||||
context = self.get_context()
|
|
||||||
zone = self.get_zone_fixture()
|
|
||||||
self.backend.infoblox.get_dns_view = MagicMock(return_value='default')
|
|
||||||
self.backend.create_zone(context, zone)
|
|
||||||
self.backend.infoblox.create_zone_auth.assert_called_once_with(
|
|
||||||
fqdn='test.example.com',
|
|
||||||
dns_view='default')
|
|
||||||
|
|
||||||
def test_update_zone(self):
|
|
||||||
self.set_up_backend()
|
|
||||||
context = self.get_context()
|
|
||||||
zone = objects.Zone().from_dict(self.get_zone_fixture())
|
|
||||||
self.backend.update_zone(context, zone)
|
|
||||||
|
|
||||||
def test_delete_zone(self):
|
|
||||||
self.set_up_backend()
|
|
||||||
context = self.get_context()
|
|
||||||
zone = self.get_zone_fixture()
|
|
||||||
self.backend.create_zone(context, zone)
|
|
||||||
self.backend.delete_zone(context, zone)
|
|
||||||
self.backend.infoblox.delete_zone_auth.assert_called_once_with(
|
|
||||||
'test.example.com')
|
|
||||||
|
|
||||||
def test_missing_wapi_url(self):
|
|
||||||
options = [{'key': 'username', 'value': 'test'},
|
|
||||||
{'key': 'password', 'value': 'test'},
|
|
||||||
{'key': 'ns_group', 'value': 'test'}]
|
|
||||||
|
|
||||||
target = self.get_target_fixture(options=options)
|
|
||||||
six.assertRaisesRegex(self, ibexceptions.InfobloxIsMisconfigured,
|
|
||||||
"wapi_url",
|
|
||||||
self.set_up_backend, target)
|
|
||||||
|
|
||||||
def test_missing_username(self):
|
|
||||||
options = [{'key': 'wapi_url', 'value': 'test'},
|
|
||||||
{'key': 'password', 'value': 'test'},
|
|
||||||
{'key': 'ns_group', 'value': 'test'}]
|
|
||||||
|
|
||||||
target = self.get_target_fixture(options=options)
|
|
||||||
six.assertRaisesRegex(self, ibexceptions.InfobloxIsMisconfigured,
|
|
||||||
"username",
|
|
||||||
self.set_up_backend, target)
|
|
||||||
|
|
||||||
def test_missing_password(self):
|
|
||||||
options = [{'key': 'wapi_url', 'value': 'test'},
|
|
||||||
{'key': 'username', 'value': 'test'},
|
|
||||||
{'key': 'ns_group', 'value': 'test'}]
|
|
||||||
|
|
||||||
target = self.get_target_fixture(options=options)
|
|
||||||
six.assertRaisesRegex(self, ibexceptions.InfobloxIsMisconfigured,
|
|
||||||
"password",
|
|
||||||
self.set_up_backend, target)
|
|
||||||
|
|
||||||
def test_missing_ns_group(self):
|
|
||||||
options = [{'key': 'wapi_url', 'value': 'test'},
|
|
||||||
{'key': 'username', 'value': 'test'},
|
|
||||||
{'key': 'password', 'value': 'test'}]
|
|
||||||
|
|
||||||
target = self.get_target_fixture(options=options)
|
|
||||||
six.assertRaisesRegex(self, ibexceptions.InfobloxIsMisconfigured,
|
|
||||||
"ns_group",
|
|
||||||
self.set_up_backend, target)
|
|
||||||
|
|
||||||
def test_wrong_port(self):
|
|
||||||
masters = [{'host': '1.1.1.1', 'port': 100}]
|
|
||||||
target = self.get_target_fixture(masters=masters)
|
|
||||||
six.assertRaisesRegex(self, ConfigurationError,
|
|
||||||
"port 53",
|
|
||||||
self.set_up_backend, target)
|
|
224
designate/tests/unit/backend/test_agent.py
Normal file
224
designate/tests/unit/backend/test_agent.py
Normal file
@ -0,0 +1,224 @@
|
|||||||
|
# Author: Federico Ceratto <federico.ceratto@hpe.com>
|
||||||
|
#
|
||||||
|
# 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 dns
|
||||||
|
import dns.rdataclass
|
||||||
|
import dns.rdatatype
|
||||||
|
import mock
|
||||||
|
|
||||||
|
import designate.backend.agent as agent
|
||||||
|
import designate.backend.private_codes as pcodes
|
||||||
|
from designate import exceptions
|
||||||
|
from designate import objects
|
||||||
|
from designate import tests
|
||||||
|
from designate.mdns import rpcapi as mdns_api
|
||||||
|
from designate.tests.unit import RoObject
|
||||||
|
|
||||||
|
|
||||||
|
class AgentBackendTestCase(tests.TestCase):
|
||||||
|
def setUp(self):
|
||||||
|
super(AgentBackendTestCase, self).setUp()
|
||||||
|
self.CONF.set_override('poll_timeout', 1, 'service:pool_manager')
|
||||||
|
self.CONF.set_override('poll_retry_interval', 4,
|
||||||
|
'service:pool_manager')
|
||||||
|
self.CONF.set_override('poll_max_retries', 5, 'service:pool_manager')
|
||||||
|
self.CONF.set_override('poll_delay', 6, 'service:pool_manager')
|
||||||
|
|
||||||
|
self.context = self.get_context()
|
||||||
|
self.zone = objects.Zone(
|
||||||
|
id='e2bed4dc-9d01-11e4-89d3-123b93f75cba',
|
||||||
|
name='example.com.',
|
||||||
|
email='example@example.com',
|
||||||
|
)
|
||||||
|
|
||||||
|
self.target = {
|
||||||
|
'id': '4588652b-50e7-46b9-b688-a9bad40a873e',
|
||||||
|
'type': 'agent',
|
||||||
|
'masters': [],
|
||||||
|
'options': [
|
||||||
|
{'key': 'host', 'value': 2},
|
||||||
|
{'key': 'port', 'value': 3},
|
||||||
|
],
|
||||||
|
}
|
||||||
|
|
||||||
|
self.backend = agent.AgentPoolBackend(
|
||||||
|
objects.PoolTarget.from_dict(self.target)
|
||||||
|
)
|
||||||
|
|
||||||
|
@mock.patch.object(mdns_api.MdnsAPI, 'get_instance')
|
||||||
|
def test_mdns_api(self, mock_get_instance):
|
||||||
|
self.assertIsInstance(self.backend.mdns_api, mock.Mock)
|
||||||
|
|
||||||
|
@mock.patch.object(mdns_api.MdnsAPI, 'get_instance')
|
||||||
|
def test_create_zone(self, mock_get_instance):
|
||||||
|
self.backend._make_and_send_dns_message = mock.Mock(
|
||||||
|
return_value=(1, 2))
|
||||||
|
|
||||||
|
out = self.backend.create_zone(self.context, self.zone)
|
||||||
|
|
||||||
|
self.backend._make_and_send_dns_message.assert_called_with(
|
||||||
|
self.zone.name, 1, 14, pcodes.CREATE, pcodes.SUCCESS, 2, 3)
|
||||||
|
self.assertIsNone(out)
|
||||||
|
|
||||||
|
@mock.patch.object(mdns_api.MdnsAPI, 'get_instance')
|
||||||
|
def test_create_zone_exception(self, mock_get_instance):
|
||||||
|
self.backend._make_and_send_dns_message = mock.Mock(
|
||||||
|
return_value=(None, 2))
|
||||||
|
|
||||||
|
self.assertRaisesRegex(
|
||||||
|
exceptions.Backend, 'create_zone.* failed',
|
||||||
|
self.backend.create_zone, self.context, self.zone,
|
||||||
|
)
|
||||||
|
|
||||||
|
self.backend._make_and_send_dns_message.assert_called_with(
|
||||||
|
self.zone.name, 1, 14, pcodes.CREATE, pcodes.SUCCESS, 2, 3)
|
||||||
|
|
||||||
|
@mock.patch.object(mdns_api.MdnsAPI, 'get_instance')
|
||||||
|
def test_update_zone(self, mock_get_instance):
|
||||||
|
self.backend.mdns_api.notify_zone_changed = mock.Mock()
|
||||||
|
|
||||||
|
out = self.backend.update_zone(self.context, self.zone)
|
||||||
|
|
||||||
|
self.backend.mdns_api.notify_zone_changed.assert_called_with(
|
||||||
|
self.context, self.zone, 2, 3, 1, 4, 5, 6)
|
||||||
|
self.assertIsNone(out)
|
||||||
|
|
||||||
|
@mock.patch.object(mdns_api.MdnsAPI, 'get_instance')
|
||||||
|
def test_delete_zone(self, mock_get_instance):
|
||||||
|
self.backend._make_and_send_dns_message = mock.Mock(
|
||||||
|
return_value=(1, 2))
|
||||||
|
|
||||||
|
out = self.backend.delete_zone(self.context, self.zone)
|
||||||
|
|
||||||
|
self.backend._make_and_send_dns_message.assert_called_with(
|
||||||
|
self.zone.name, 1, 14, pcodes.DELETE, pcodes.SUCCESS, 2, 3)
|
||||||
|
self.assertIsNone(out)
|
||||||
|
|
||||||
|
@mock.patch.object(mdns_api.MdnsAPI, 'get_instance')
|
||||||
|
def test_delete_zone_exception(self, mock_get_instance):
|
||||||
|
self.backend._make_and_send_dns_message = mock.Mock(
|
||||||
|
return_value=(None, 2))
|
||||||
|
|
||||||
|
self.assertRaisesRegex(
|
||||||
|
exceptions.Backend, 'failed delete_zone',
|
||||||
|
self.backend.delete_zone, self.context, self.zone,
|
||||||
|
)
|
||||||
|
|
||||||
|
self.backend._make_and_send_dns_message.assert_called_with(
|
||||||
|
self.zone.name, 1, 14, pcodes.DELETE, pcodes.SUCCESS, 2, 3)
|
||||||
|
|
||||||
|
def test_make_and_send_dns_message_timeout(self):
|
||||||
|
self.backend._make_dns_message = mock.Mock(return_value='')
|
||||||
|
self.backend._send_dns_message = mock.Mock(
|
||||||
|
return_value=dns.exception.Timeout())
|
||||||
|
|
||||||
|
out = self.backend._make_and_send_dns_message('h', 123, 1, 2, 3, 4, 5)
|
||||||
|
|
||||||
|
self.assertEqual((None, 0), out)
|
||||||
|
|
||||||
|
def test_make_and_send_dns_message_bad_response(self):
|
||||||
|
self.backend._make_dns_message = mock.Mock(return_value='')
|
||||||
|
self.backend._send_dns_message = mock.Mock(
|
||||||
|
return_value=agent.dns_query.BadResponse())
|
||||||
|
|
||||||
|
out = self.backend._make_and_send_dns_message('h', 123, 1, 2, 3, 4, 5)
|
||||||
|
|
||||||
|
self.assertEqual((None, 0), out)
|
||||||
|
|
||||||
|
def test_make_and_send_dns_message_missing_AA_flags(self):
|
||||||
|
self.backend._make_dns_message = mock.Mock(return_value='')
|
||||||
|
response = RoObject(
|
||||||
|
rcode=mock.Mock(return_value=dns.rcode.NOERROR),
|
||||||
|
# rcode is NOERROR but (flags & dns.flags.AA) gives 0
|
||||||
|
flags=0,
|
||||||
|
)
|
||||||
|
self.backend._send_dns_message = mock.Mock(return_value=response)
|
||||||
|
|
||||||
|
out = self.backend._make_and_send_dns_message('h', 123, 1, 2, 3, 4, 5)
|
||||||
|
|
||||||
|
self.assertEqual((None, 0), out)
|
||||||
|
|
||||||
|
def test_make_and_send_dns_message_error_flags(self):
|
||||||
|
self.backend._make_dns_message = mock.Mock(return_value='')
|
||||||
|
response = RoObject(
|
||||||
|
rcode=mock.Mock(return_value=dns.rcode.NOERROR),
|
||||||
|
# rcode is NOERROR but flags are not NOERROR
|
||||||
|
flags=123,
|
||||||
|
ednsflags=321
|
||||||
|
)
|
||||||
|
self.backend._send_dns_message = mock.Mock(return_value=response)
|
||||||
|
|
||||||
|
out = self.backend._make_and_send_dns_message('h', 123, 1, 2, 3, 4, 5)
|
||||||
|
|
||||||
|
self.assertEqual((None, 0), out)
|
||||||
|
|
||||||
|
def test_make_and_send_dns_message(self):
|
||||||
|
self.backend._make_dns_message = mock.Mock(return_value='')
|
||||||
|
response = RoObject(
|
||||||
|
rcode=mock.Mock(return_value=dns.rcode.NOERROR),
|
||||||
|
flags=agent.dns.flags.AA,
|
||||||
|
ednsflags=321
|
||||||
|
)
|
||||||
|
self.backend._send_dns_message = mock.Mock(return_value=response)
|
||||||
|
|
||||||
|
out = self.backend._make_and_send_dns_message('h', 123, 1, 2, 3, 4, 5)
|
||||||
|
|
||||||
|
self.assertEqual((response, 0), out)
|
||||||
|
|
||||||
|
@mock.patch.object(agent.dns_query, 'tcp')
|
||||||
|
@mock.patch.object(agent.dns_query, 'udp')
|
||||||
|
def test_send_dns_message(self, mock_udp, mock_tcp):
|
||||||
|
mock_udp.return_value = 'mock udp resp'
|
||||||
|
|
||||||
|
out = self.backend._send_dns_message('msg', 'host', 123, 1)
|
||||||
|
|
||||||
|
self.assertFalse(agent.dns_query.tcp.called)
|
||||||
|
agent.dns_query.udp.assert_called_with('msg', 'host', port=123,
|
||||||
|
timeout=1)
|
||||||
|
self.assertEqual('mock udp resp', out)
|
||||||
|
|
||||||
|
@mock.patch.object(agent.dns_query, 'tcp')
|
||||||
|
@mock.patch.object(agent.dns_query, 'udp')
|
||||||
|
def test_send_dns_message_timeout(self, mock_udp, mock_tcp):
|
||||||
|
mock_udp.side_effect = dns.exception.Timeout
|
||||||
|
|
||||||
|
out = self.backend._send_dns_message('msg', 'host', 123, 1)
|
||||||
|
|
||||||
|
agent.dns_query.udp.assert_called_with('msg', 'host', port=123,
|
||||||
|
timeout=1)
|
||||||
|
self.assertIsInstance(out, dns.exception.Timeout)
|
||||||
|
|
||||||
|
@mock.patch.object(agent.dns_query, 'tcp')
|
||||||
|
@mock.patch.object(agent.dns_query, 'udp')
|
||||||
|
def test_send_dns_message_bad_response(self, mock_udp, mock_tcp):
|
||||||
|
mock_udp.side_effect = agent.dns_query.BadResponse
|
||||||
|
|
||||||
|
out = self.backend._send_dns_message('msg', 'host', 123, 1)
|
||||||
|
|
||||||
|
agent.dns_query.udp.assert_called_with('msg', 'host', port=123,
|
||||||
|
timeout=1)
|
||||||
|
self.assertIsInstance(out, agent.dns_query.BadResponse)
|
||||||
|
|
||||||
|
@mock.patch.object(agent.dns_query, 'tcp')
|
||||||
|
@mock.patch.object(agent.dns_query, 'udp')
|
||||||
|
def test_send_dns_message_tcp(self, mock_udp, mock_tcp):
|
||||||
|
self.CONF.set_override('all_tcp', True, 'service:mdns')
|
||||||
|
|
||||||
|
mock_tcp.return_value = 'mock tcp resp'
|
||||||
|
|
||||||
|
out = self.backend._send_dns_message('msg', 'host', 123, 1)
|
||||||
|
|
||||||
|
self.assertFalse(agent.dns_query.udp.called)
|
||||||
|
agent.dns_query.tcp.assert_called_with('msg', 'host', port=123,
|
||||||
|
timeout=1)
|
||||||
|
self.assertEqual('mock tcp resp', out)
|
91
designate/tests/unit/backend/test_designate.py
Normal file
91
designate/tests/unit/backend/test_designate.py
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
# Copyright 2015 Hewlett-Packard Development Company, L.P.
|
||||||
|
#
|
||||||
|
# Author: Endre Karlson <endre.karlson@hpe.com>
|
||||||
|
#
|
||||||
|
# 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 testtools
|
||||||
|
from designateclient import exceptions
|
||||||
|
from mock import NonCallableMagicMock
|
||||||
|
from mock import patch
|
||||||
|
from oslo_log import log as logging
|
||||||
|
|
||||||
|
from designate import objects
|
||||||
|
from designate import tests
|
||||||
|
from designate.backend import impl_designate
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class DesignateBackendTestCase(tests.TestCase):
|
||||||
|
def setUp(self):
|
||||||
|
super(DesignateBackendTestCase, self).setUp()
|
||||||
|
self.zone = objects.Zone(
|
||||||
|
id='e2bed4dc-9d01-11e4-89d3-123b93f75cba',
|
||||||
|
name='example.com.',
|
||||||
|
email='example@example.com',
|
||||||
|
)
|
||||||
|
|
||||||
|
self.target = {
|
||||||
|
'id': '4588652b-50e7-46b9-b688-a9bad40a873e',
|
||||||
|
'type': 'designate',
|
||||||
|
'masters': [
|
||||||
|
{'host': '192.0.2.1', 'port': 53},
|
||||||
|
],
|
||||||
|
'options': [
|
||||||
|
{'key': 'username', 'value': 'user'},
|
||||||
|
{'key': 'password', 'value': 'secret'},
|
||||||
|
{'key': 'project_name', 'value': 'project'},
|
||||||
|
{'key': 'project_zone_name', 'value': 'project_zone'},
|
||||||
|
{'key': 'user_zone_name', 'value': 'user_zone'},
|
||||||
|
],
|
||||||
|
}
|
||||||
|
|
||||||
|
self.backend = impl_designate.DesignateBackend(
|
||||||
|
objects.PoolTarget.from_dict(self.target)
|
||||||
|
)
|
||||||
|
|
||||||
|
# Mock client
|
||||||
|
self.client = NonCallableMagicMock()
|
||||||
|
zones = NonCallableMagicMock(spec_set=[
|
||||||
|
'create', 'delete'])
|
||||||
|
|
||||||
|
self.client.configure_mock(zones=zones)
|
||||||
|
|
||||||
|
def test_create_zone(self):
|
||||||
|
masters = ["%(host)s:%(port)s" % self.target['masters'][0]]
|
||||||
|
with patch.object(self.backend, '_get_client',
|
||||||
|
return_value=self.client):
|
||||||
|
self.backend.create_zone(self.admin_context, self.zone)
|
||||||
|
self.client.zones.create.assert_called_once_with(
|
||||||
|
self.zone.name, 'SECONDARY', masters=masters)
|
||||||
|
|
||||||
|
def test_delete_zone(self):
|
||||||
|
with patch.object(self.backend, '_get_client',
|
||||||
|
return_value=self.client):
|
||||||
|
self.backend.delete_zone(self.admin_context, self.zone)
|
||||||
|
self.client.zones.delete.assert_called_once_with(self.zone.name)
|
||||||
|
|
||||||
|
def test_delete_zone_notfound(self):
|
||||||
|
self.client.delete.side_effect = exceptions.NotFound
|
||||||
|
with patch.object(self.backend, '_get_client',
|
||||||
|
return_value=self.client):
|
||||||
|
self.backend.delete_zone(self.admin_context, self.zone)
|
||||||
|
self.client.zones.delete.assert_called_once_with(self.zone.name)
|
||||||
|
|
||||||
|
def test_delete_zone_exc(self):
|
||||||
|
self.client.zones.delete.side_effect = Exception
|
||||||
|
with testtools.ExpectedException(Exception):
|
||||||
|
with patch.object(self.backend, '_get_client',
|
||||||
|
return_value=self.client):
|
||||||
|
self.backend.delete_zone(self.admin_context, self.zone)
|
||||||
|
self.client.zones.delete.assert_called_once_with(self.zone.name)
|
@ -13,19 +13,15 @@
|
|||||||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
# License for the specific language governing permissions and limitations
|
# License for the specific language governing permissions and limitations
|
||||||
# under the License.
|
# under the License.
|
||||||
|
import requests_mock
|
||||||
from oslo_serialization import jsonutils
|
|
||||||
from requests_mock.contrib import fixture as req_fixture
|
|
||||||
import testtools
|
|
||||||
|
|
||||||
from designate import objects
|
from designate import objects
|
||||||
|
from designate import tests
|
||||||
from designate.backend import impl_dynect
|
from designate.backend import impl_dynect
|
||||||
from designate.tests.test_backend import BackendTestCase
|
|
||||||
|
|
||||||
MASTERS = ["10.0.0.1"]
|
MASTERS = ["10.0.0.1"]
|
||||||
CONTACT = 'jdoe@myco.biz'
|
CONTACT = 'jdoe@myco.biz'
|
||||||
|
|
||||||
|
|
||||||
LOGIN_SUCCESS = {
|
LOGIN_SUCCESS = {
|
||||||
"status": "success",
|
"status": "success",
|
||||||
"data": {
|
"data": {
|
||||||
@ -93,7 +89,6 @@ TARGET_EXISTS = {
|
|||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
ACTIVATE_SUCCESS = {
|
ACTIVATE_SUCCESS = {
|
||||||
"status": "success",
|
"status": "success",
|
||||||
"data": {
|
"data": {
|
||||||
@ -115,73 +110,74 @@ ACTIVATE_SUCCESS = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class DynECTTestsCase(BackendTestCase):
|
class DynECTTestsCase(tests.TestCase):
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
super(DynECTTestsCase, self).setUp()
|
super(DynECTTestsCase, self).setUp()
|
||||||
self.target = objects.PoolTarget.from_dict({
|
|
||||||
|
self.base_address = 'https://api.dynect.net:443/REST'
|
||||||
|
self.context = self.get_context()
|
||||||
|
self.zone = objects.Zone(
|
||||||
|
id='e2bed4dc-9d01-11e4-89d3-123b93f75cba',
|
||||||
|
name='example.com.',
|
||||||
|
email='example@example.com',
|
||||||
|
)
|
||||||
|
|
||||||
|
self.target = {
|
||||||
'id': '4588652b-50e7-46b9-b688-a9bad40a873e',
|
'id': '4588652b-50e7-46b9-b688-a9bad40a873e',
|
||||||
'type': 'dyndns',
|
'type': 'dyndns',
|
||||||
'masters': [{'host': '192.0.2.1', 'port': 53}],
|
'masters': [
|
||||||
|
{'host': '192.0.2.1', 'port': 53}
|
||||||
|
],
|
||||||
'options': [
|
'options': [
|
||||||
{'key': 'username', 'value': 'example'},
|
{'key': 'username', 'value': 'example'},
|
||||||
{'key': 'password', 'value': 'secret'},
|
{'key': 'password', 'value': 'secret'},
|
||||||
{'key': 'customer_name', 'value': 'customer'}],
|
{'key': 'customer_name', 'value': 'customer'}],
|
||||||
})
|
}
|
||||||
|
|
||||||
self.backend = impl_dynect.DynECTBackend(self.target)
|
self.backend = impl_dynect.DynECTBackend(
|
||||||
self.requests = self.useFixture(req_fixture.Fixture())
|
objects.PoolTarget.from_dict(self.target)
|
||||||
|
)
|
||||||
|
|
||||||
def stub_url(self, method, parts=None, base_url=None, json=None, **kwargs):
|
@requests_mock.mock()
|
||||||
if not base_url:
|
def test_create_zone_raise_dynclienterror(self, req_mock):
|
||||||
base_url = 'https://api.dynect.net:443/REST'
|
# https://api.dynect.net:443/REST/Session
|
||||||
|
req_mock.post(
|
||||||
|
'%s/Session' % self.base_address,
|
||||||
|
json=LOGIN_SUCCESS,
|
||||||
|
)
|
||||||
|
|
||||||
if json:
|
req_mock.post(
|
||||||
kwargs['text'] = jsonutils.dumps(json)
|
'%s/Secondary/example.com' % self.base_address,
|
||||||
headers = kwargs.setdefault('headers', {})
|
|
||||||
headers['Content-Type'] = 'application/json'
|
|
||||||
|
|
||||||
if parts:
|
|
||||||
url = '/'.join([p.strip('/') for p in [base_url] + parts])
|
|
||||||
else:
|
|
||||||
url = base_url
|
|
||||||
|
|
||||||
url = url.replace("/?", "?")
|
|
||||||
|
|
||||||
return self.requests.register_uri(method, url, **kwargs)
|
|
||||||
|
|
||||||
def _stub_login(self):
|
|
||||||
self.stub_url('POST', ['/Session'], json=LOGIN_SUCCESS)
|
|
||||||
self.stub_url('DELETE', ['/Session'], json=LOGIN_SUCCESS)
|
|
||||||
|
|
||||||
def test_create_zone_raise_dynclienterror(self):
|
|
||||||
context = self.get_context()
|
|
||||||
zone = self.create_zone()
|
|
||||||
|
|
||||||
self._stub_login()
|
|
||||||
|
|
||||||
self.stub_url(
|
|
||||||
'POST', ['/Secondary/example.com'],
|
|
||||||
json=INVALID_MASTER_DATA,
|
json=INVALID_MASTER_DATA,
|
||||||
status_code=400)
|
status_code=400,
|
||||||
|
)
|
||||||
|
|
||||||
with testtools.ExpectedException(impl_dynect.DynClientError):
|
self.assertRaisesRegex(
|
||||||
self.backend.create_zone(context, zone)
|
impl_dynect.DynClientError, 'Zone not created',
|
||||||
|
self.backend.create_zone, self.context, self.zone,
|
||||||
|
)
|
||||||
|
|
||||||
def test_create_zone_duplicate_updates_existing(self):
|
@requests_mock.mock()
|
||||||
context = self.get_context()
|
def test_create_zone_duplicate_updates_existing(self, req_mock):
|
||||||
zone = self.create_zone()
|
req_mock.post(
|
||||||
|
'%s/Session' % self.base_address,
|
||||||
|
json=LOGIN_SUCCESS,
|
||||||
|
)
|
||||||
|
|
||||||
self._stub_login()
|
req_mock.delete(
|
||||||
|
'%s/Session' % self.base_address,
|
||||||
|
json=LOGIN_SUCCESS,
|
||||||
|
)
|
||||||
|
|
||||||
parts = ['/Secondary', '/%s' % zone['name'].rstrip('.')]
|
req_mock.post(
|
||||||
|
'%s/Secondary/example.com' % self.base_address,
|
||||||
self.stub_url(
|
|
||||||
'POST', parts,
|
|
||||||
json=TARGET_EXISTS,
|
json=TARGET_EXISTS,
|
||||||
status_code=400)
|
status_code=400,
|
||||||
|
)
|
||||||
|
|
||||||
update = self.stub_url('PUT', parts, json=ACTIVATE_SUCCESS)
|
req_mock.put(
|
||||||
|
'%s/Secondary/example.com' % self.base_address,
|
||||||
|
json=ACTIVATE_SUCCESS,
|
||||||
|
)
|
||||||
|
|
||||||
self.backend.create_zone(context, zone)
|
self.backend.create_zone(self.context, self.zone)
|
||||||
|
|
||||||
self.assertTrue(update.called)
|
|
164
designate/tests/unit/backend/test_infoblox.py
Normal file
164
designate/tests/unit/backend/test_infoblox.py
Normal file
@ -0,0 +1,164 @@
|
|||||||
|
# Copyright 2015 Infoblox Inc.
|
||||||
|
# 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.
|
||||||
|
import mock
|
||||||
|
import requests_mock
|
||||||
|
|
||||||
|
from designate import exceptions
|
||||||
|
from designate import objects
|
||||||
|
from designate import tests
|
||||||
|
from designate.backend import impl_infoblox
|
||||||
|
from designate.backend.impl_infoblox import ibexceptions
|
||||||
|
from designate.mdns import rpcapi as mdns_rpcapi
|
||||||
|
|
||||||
|
|
||||||
|
class InfobloxBackendTestCase(tests.TestCase):
|
||||||
|
def setUp(self):
|
||||||
|
super(InfobloxBackendTestCase, self).setUp()
|
||||||
|
self.base_address = 'https://localhost/wapi'
|
||||||
|
|
||||||
|
self.context = self.get_context()
|
||||||
|
self.zone = objects.Zone(
|
||||||
|
id='e2bed4dc-9d01-11e4-89d3-123b93f75cba',
|
||||||
|
name='example.com.',
|
||||||
|
email='example@example.com',
|
||||||
|
)
|
||||||
|
|
||||||
|
self.target = {
|
||||||
|
'id': '4588652b-50e7-46b9-b688-a9bad40a873e',
|
||||||
|
'type': 'infoblox',
|
||||||
|
'masters': [
|
||||||
|
{'host': '1.1.1.1', 'port': 53},
|
||||||
|
],
|
||||||
|
'options': [
|
||||||
|
{'key': 'wapi_url', 'value': 'https://localhost/wapi/v2.0/'},
|
||||||
|
{'key': 'username', 'value': 'test'},
|
||||||
|
{'key': 'password', 'value': 'test'},
|
||||||
|
{'key': 'ns_group', 'value': 'test'},
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
self.backend = impl_infoblox.InfobloxBackend(
|
||||||
|
objects.PoolTarget.from_dict(self.target)
|
||||||
|
)
|
||||||
|
|
||||||
|
@requests_mock.mock()
|
||||||
|
def test_create_zone(self, req_mock):
|
||||||
|
req_mock.post(
|
||||||
|
'%s/v2.0/zone_auth' % self.base_address,
|
||||||
|
json={},
|
||||||
|
)
|
||||||
|
|
||||||
|
req_mock.get(
|
||||||
|
'%s/v2.0/zone_auth' % self.base_address,
|
||||||
|
json={},
|
||||||
|
)
|
||||||
|
|
||||||
|
self.backend.create_zone(self.context, self.zone)
|
||||||
|
|
||||||
|
@mock.patch.object(mdns_rpcapi.MdnsAPI, 'notify_zone_changed')
|
||||||
|
def test_update_zone(self, mock_notify_zone_changed):
|
||||||
|
self.backend.update_zone(self.context, self.zone)
|
||||||
|
|
||||||
|
mock_notify_zone_changed.assert_called_with(
|
||||||
|
self.context, self.zone, '127.0.0.1', 53, 30, 15, 10, 5)
|
||||||
|
|
||||||
|
@requests_mock.mock()
|
||||||
|
def test_delete_zone(self, req_mock):
|
||||||
|
req_mock.post(
|
||||||
|
'%s/v2.0/zone_auth' % self.base_address,
|
||||||
|
json={},
|
||||||
|
)
|
||||||
|
|
||||||
|
req_mock.get(
|
||||||
|
'%s/v2.0/zone_auth' % self.base_address,
|
||||||
|
json={},
|
||||||
|
)
|
||||||
|
|
||||||
|
self.backend.create_zone(self.context, self.zone)
|
||||||
|
self.backend.delete_zone(self.context, self.zone)
|
||||||
|
|
||||||
|
def test_missing_wapi_url(self):
|
||||||
|
target = dict(self.target)
|
||||||
|
target['options'] = [
|
||||||
|
{'key': 'username', 'value': 'test'},
|
||||||
|
{'key': 'password', 'value': 'test'},
|
||||||
|
{'key': 'ns_group', 'value': 'test'},
|
||||||
|
]
|
||||||
|
|
||||||
|
pool_target = objects.PoolTarget.from_dict(target)
|
||||||
|
|
||||||
|
self.assertRaisesRegex(
|
||||||
|
ibexceptions.InfobloxIsMisconfigured, "wapi_url",
|
||||||
|
impl_infoblox.InfobloxBackend, pool_target,
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_missing_username(self):
|
||||||
|
target = dict(self.target)
|
||||||
|
target['options'] = [
|
||||||
|
{'key': 'wapi_url', 'value': 'test'},
|
||||||
|
{'key': 'password', 'value': 'test'},
|
||||||
|
{'key': 'ns_group', 'value': 'test'}
|
||||||
|
]
|
||||||
|
|
||||||
|
pool_target = objects.PoolTarget.from_dict(target)
|
||||||
|
|
||||||
|
self.assertRaisesRegex(
|
||||||
|
ibexceptions.InfobloxIsMisconfigured, "username",
|
||||||
|
impl_infoblox.InfobloxBackend, pool_target,
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_missing_password(self):
|
||||||
|
target = dict(self.target)
|
||||||
|
target['options'] = [
|
||||||
|
{'key': 'wapi_url', 'value': 'test'},
|
||||||
|
{'key': 'username', 'value': 'test'},
|
||||||
|
{'key': 'ns_group', 'value': 'test'},
|
||||||
|
]
|
||||||
|
|
||||||
|
pool_target = objects.PoolTarget.from_dict(target)
|
||||||
|
|
||||||
|
self.assertRaisesRegex(
|
||||||
|
ibexceptions.InfobloxIsMisconfigured, "password",
|
||||||
|
impl_infoblox.InfobloxBackend, pool_target,
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_missing_ns_group(self):
|
||||||
|
target = dict(self.target)
|
||||||
|
target['options'] = [
|
||||||
|
{'key': 'wapi_url', 'value': 'test'},
|
||||||
|
{'key': 'username', 'value': 'test'},
|
||||||
|
{'key': 'password', 'value': 'test'},
|
||||||
|
]
|
||||||
|
|
||||||
|
pool_target = objects.PoolTarget.from_dict(target)
|
||||||
|
|
||||||
|
self.assertRaisesRegex(
|
||||||
|
ibexceptions.InfobloxIsMisconfigured, "ns_group",
|
||||||
|
impl_infoblox.InfobloxBackend, pool_target,
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_wrong_port(self):
|
||||||
|
target = dict(self.target)
|
||||||
|
target['masters'] = [
|
||||||
|
{'host': '1.1.1.1', 'port': 100},
|
||||||
|
]
|
||||||
|
|
||||||
|
pool_target = objects.PoolTarget.from_dict(target)
|
||||||
|
|
||||||
|
self.assertRaisesRegex(
|
||||||
|
exceptions.ConfigurationError,
|
||||||
|
'Infoblox only supports mDNS instances on port 53',
|
||||||
|
impl_infoblox.InfobloxBackend, pool_target,
|
||||||
|
)
|
@ -13,7 +13,6 @@
|
|||||||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
# License for the specific language governing permissions and limitations
|
# License for the specific language governing permissions and limitations
|
||||||
# under the License.
|
# under the License.
|
||||||
|
|
||||||
import socket
|
import socket
|
||||||
import ssl
|
import ssl
|
||||||
|
|
||||||
@ -22,34 +21,43 @@ import mock
|
|||||||
|
|
||||||
from designate import exceptions
|
from designate import exceptions
|
||||||
from designate import objects
|
from designate import objects
|
||||||
from designate.tests.test_backend import BackendTestCase
|
from designate import tests
|
||||||
from designate.backend import impl_nsd4
|
from designate.backend import impl_nsd4
|
||||||
|
|
||||||
|
|
||||||
# NOTE: We'll only test the specifics to the nsd4 backend here.
|
class NSD4BackendTestCase(tests.TestCase):
|
||||||
# Rest is handled via scenarios
|
|
||||||
class NSD4BackendTestCase(BackendTestCase):
|
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
super(NSD4BackendTestCase, self).setUp()
|
super(NSD4BackendTestCase, self).setUp()
|
||||||
|
|
||||||
# NOTE(hieulq): we mock out NSD4 back-end with random port
|
|
||||||
|
|
||||||
keyfile = mock.sentinel.key
|
keyfile = mock.sentinel.key
|
||||||
certfile = mock.sentinel.cert
|
certfile = mock.sentinel.cert
|
||||||
|
|
||||||
|
self.context = self.get_context()
|
||||||
|
self.zone = objects.Zone(
|
||||||
|
id='e2bed4dc-9d01-11e4-89d3-123b93f75cba',
|
||||||
|
name='example.com.',
|
||||||
|
email='example@example.com',
|
||||||
|
)
|
||||||
|
|
||||||
self.port = 6969
|
self.port = 6969
|
||||||
self.target = objects.PoolTarget.from_dict({
|
self.target = {
|
||||||
'id': '4588652b-50e7-46b9-b688-a9bad40a873e',
|
'id': '4588652b-50e7-46b9-b688-a9bad40a873e',
|
||||||
'type': 'nsd4',
|
'type': 'nsd4',
|
||||||
'masters': [{'host': '192.0.2.1', 'port': 53},
|
'masters': [
|
||||||
{'host': '192.0.2.2', 'port': 35}],
|
{'host': '192.0.2.1', 'port': 53},
|
||||||
|
{'host': '192.0.2.2', 'port': 35},
|
||||||
|
],
|
||||||
'options': [
|
'options': [
|
||||||
{'key': 'keyfile', 'value': keyfile.name},
|
{'key': 'keyfile', 'value': keyfile.name},
|
||||||
{'key': 'certfile', 'value': certfile.name},
|
{'key': 'certfile', 'value': certfile.name},
|
||||||
{'key': 'pattern', 'value': 'test-pattern'},
|
{'key': 'pattern', 'value': 'test-pattern'},
|
||||||
{'key': 'port', 'value': str(self.port)}
|
{'key': 'port', 'value': str(self.port)},
|
||||||
],
|
],
|
||||||
})
|
}
|
||||||
self.backend = impl_nsd4.NSD4Backend(self.target)
|
|
||||||
|
self.backend = impl_nsd4.NSD4Backend(
|
||||||
|
objects.PoolTarget.from_dict(self.target)
|
||||||
|
)
|
||||||
|
|
||||||
@mock.patch.object(eventlet, 'connect')
|
@mock.patch.object(eventlet, 'connect')
|
||||||
@mock.patch.object(eventlet, 'wrap_ssl')
|
@mock.patch.object(eventlet, 'wrap_ssl')
|
||||||
@ -64,20 +72,17 @@ class NSD4BackendTestCase(BackendTestCase):
|
|||||||
else:
|
else:
|
||||||
stream.read.return_value = 'ok'
|
stream.read.return_value = 'ok'
|
||||||
|
|
||||||
context = self.get_context()
|
|
||||||
zone = self.get_zone_fixture()
|
|
||||||
|
|
||||||
if command_context is 'create':
|
if command_context is 'create':
|
||||||
self.backend.create_zone(context, zone)
|
self.backend.create_zone(self.context, self.zone)
|
||||||
command = 'NSDCT1 addzone %s test-pattern\n' % zone['name']
|
command = 'NSDCT1 addzone %s test-pattern\n' % self.zone.name
|
||||||
elif command_context is 'delete':
|
elif command_context is 'delete':
|
||||||
self.backend.delete_zone(context, zone)
|
self.backend.delete_zone(self.context, self.zone)
|
||||||
command = 'NSDCT1 delzone %s\n' % zone['name']
|
command = 'NSDCT1 delzone %s\n' % self.zone.name
|
||||||
elif command_context is 'create_fail':
|
elif command_context is 'create_fail':
|
||||||
self.assertRaises(exceptions.Backend,
|
self.assertRaises(exceptions.Backend,
|
||||||
self.backend.create_zone,
|
self.backend.create_zone,
|
||||||
context, zone)
|
self.context, self.zone)
|
||||||
command = 'NSDCT1 addzone %s test-pattern\n' % zone['name']
|
command = 'NSDCT1 addzone %s test-pattern\n' % self.zone.name
|
||||||
|
|
||||||
stream.write.assert_called_once_with(command)
|
stream.write.assert_called_once_with(command)
|
||||||
mock_ssl.assert_called_once_with(mock.sentinel.client,
|
mock_ssl.assert_called_once_with(mock.sentinel.client,
|
||||||
@ -101,16 +106,12 @@ class NSD4BackendTestCase(BackendTestCase):
|
|||||||
|
|
||||||
def test_ssl_error(self):
|
def test_ssl_error(self):
|
||||||
self.backend._command = mock.MagicMock(side_effect=ssl.SSLError)
|
self.backend._command = mock.MagicMock(side_effect=ssl.SSLError)
|
||||||
context = self.get_context()
|
|
||||||
zone = self.get_zone_fixture()
|
|
||||||
self.assertRaises(exceptions.Backend,
|
self.assertRaises(exceptions.Backend,
|
||||||
self.backend.create_zone,
|
self.backend.create_zone,
|
||||||
context, zone)
|
self.context, self.zone)
|
||||||
|
|
||||||
def test_socket_error(self):
|
def test_socket_error(self):
|
||||||
self.backend._command = mock.MagicMock(side_effect=socket.error)
|
self.backend._command = mock.MagicMock(side_effect=socket.error)
|
||||||
context = self.get_context()
|
|
||||||
zone = self.get_zone_fixture()
|
|
||||||
self.assertRaises(exceptions.Backend,
|
self.assertRaises(exceptions.Backend,
|
||||||
self.backend.create_zone,
|
self.backend.create_zone,
|
||||||
context, zone)
|
self.context, self.zone)
|
@ -14,24 +14,25 @@ import requests_mock
|
|||||||
|
|
||||||
from designate import exceptions
|
from designate import exceptions
|
||||||
from designate import objects
|
from designate import objects
|
||||||
|
from designate import tests
|
||||||
from designate.backend import impl_pdns4
|
from designate.backend import impl_pdns4
|
||||||
from designate.mdns import rpcapi as mdns_rpcapi
|
from designate.mdns import rpcapi as mdns_rpcapi
|
||||||
from designate.tests import fixtures
|
from designate.tests import fixtures
|
||||||
from designate.tests.test_backend import BackendTestCase
|
|
||||||
|
|
||||||
|
|
||||||
class PDNS4BackendTestCase(BackendTestCase):
|
class PDNS4BackendTestCase(tests.TestCase):
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
super(PDNS4BackendTestCase, self).setUp()
|
super(PDNS4BackendTestCase, self).setUp()
|
||||||
self.stdlog = fixtures.StandardLogging()
|
self.stdlog = fixtures.StandardLogging()
|
||||||
self.useFixture(self.stdlog)
|
self.useFixture(self.stdlog)
|
||||||
|
|
||||||
self.base_address = 'http://localhost:8081/api/v1/servers'
|
self.base_address = 'http://localhost:8081/api/v1/servers'
|
||||||
|
|
||||||
self.context = self.get_context()
|
self.context = self.get_context()
|
||||||
self.zone = objects.Zone(id='e2bed4dc-9d01-11e4-89d3-123b93f75cba',
|
self.zone = objects.Zone(
|
||||||
name='example.com.',
|
id='e2bed4dc-9d01-11e4-89d3-123b93f75cba',
|
||||||
email='example@example.com')
|
name='example.com.',
|
||||||
|
email='example@example.com',
|
||||||
|
)
|
||||||
|
|
||||||
self.target = {
|
self.target = {
|
||||||
'id': '4588652b-50e7-46b9-b688-a9bad40a873e',
|
'id': '4588652b-50e7-46b9-b688-a9bad40a873e',
|
||||||
@ -53,7 +54,6 @@ class PDNS4BackendTestCase(BackendTestCase):
|
|||||||
@requests_mock.mock()
|
@requests_mock.mock()
|
||||||
@mock.patch.object(mdns_rpcapi.MdnsAPI, 'notify_zone_changed')
|
@mock.patch.object(mdns_rpcapi.MdnsAPI, 'notify_zone_changed')
|
||||||
def test_create_zone_success(self, req_mock, mock_notify_zone_changed):
|
def test_create_zone_success(self, req_mock, mock_notify_zone_changed):
|
||||||
|
|
||||||
req_mock.post(
|
req_mock.post(
|
||||||
'%s/localhost/zones' % self.base_address,
|
'%s/localhost/zones' % self.base_address,
|
||||||
)
|
)
|
@ -1,212 +0,0 @@
|
|||||||
|
|
||||||
# Author: Federico Ceratto <federico.ceratto@hpe.com>
|
|
||||||
#
|
|
||||||
# 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-test backend agent
|
|
||||||
"""
|
|
||||||
|
|
||||||
from mock import MagicMock
|
|
||||||
from mock import Mock
|
|
||||||
from mock import patch
|
|
||||||
from oslotest import base
|
|
||||||
import dns
|
|
||||||
import dns.rdataclass
|
|
||||||
import dns.rdatatype
|
|
||||||
import mock
|
|
||||||
import testtools
|
|
||||||
|
|
||||||
from designate import exceptions
|
|
||||||
from designate.tests.unit import RoObject
|
|
||||||
import designate.backend.agent as agent
|
|
||||||
import designate.backend.private_codes as pcodes
|
|
||||||
|
|
||||||
|
|
||||||
class SCAgentPoolBackend(agent.AgentPoolBackend):
|
|
||||||
def __init__(self):
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
@mock.patch.object(agent.mdns_api.MdnsAPI, 'get_instance')
|
|
||||||
@patch.object(agent.base.Backend, '__init__')
|
|
||||||
class BackendAgentTest(base.BaseTestCase):
|
|
||||||
|
|
||||||
def setUp(self, *mocks):
|
|
||||||
super(BackendAgentTest, self).setUp()
|
|
||||||
agent.CONF = RoObject({
|
|
||||||
'service:mdns': RoObject(all_tcp=False)
|
|
||||||
})
|
|
||||||
self.agent = SCAgentPoolBackend()
|
|
||||||
self.agent.timeout = 1
|
|
||||||
self.agent.host = 2
|
|
||||||
self.agent.port = 3
|
|
||||||
self.agent.retry_interval = 4
|
|
||||||
self.agent.max_retries = 5
|
|
||||||
self.agent.delay = 6
|
|
||||||
|
|
||||||
def test_mdns_api(self, *mock):
|
|
||||||
assert isinstance(self.agent.mdns_api, MagicMock)
|
|
||||||
|
|
||||||
def test_create_zone(self, *mock):
|
|
||||||
self.agent._make_and_send_dns_message = Mock(return_value=(1, 2))
|
|
||||||
|
|
||||||
out = self.agent.create_zone('ctx', RoObject(name='zn'))
|
|
||||||
|
|
||||||
self.agent._make_and_send_dns_message.assert_called_with(
|
|
||||||
'zn', 1, 14, pcodes.CREATE, pcodes.SUCCESS, 2, 3)
|
|
||||||
self.assertIsNone(out)
|
|
||||||
|
|
||||||
def test_create_zone_exception(self, *mock):
|
|
||||||
self.agent._make_and_send_dns_message = Mock(return_value=(None, 2))
|
|
||||||
|
|
||||||
with testtools.ExpectedException(exceptions.Backend):
|
|
||||||
self.agent.create_zone('ctx', RoObject(name='zn'))
|
|
||||||
|
|
||||||
self.agent._make_and_send_dns_message.assert_called_with(
|
|
||||||
'zn', 1, 14, pcodes.CREATE, pcodes.SUCCESS, 2, 3)
|
|
||||||
|
|
||||||
def test_update_zone(self, *mock):
|
|
||||||
self.agent.mdns_api.notify_zone_changed = Mock()
|
|
||||||
zone = RoObject(name='zn')
|
|
||||||
|
|
||||||
out = self.agent.update_zone('ctx', zone)
|
|
||||||
|
|
||||||
self.agent.mdns_api.notify_zone_changed.assert_called_with(
|
|
||||||
'ctx', zone, 2, 3, 1, 4, 5, 6)
|
|
||||||
self.assertIsNone(out)
|
|
||||||
|
|
||||||
def test_delete_zone(self, *mock):
|
|
||||||
self.agent._make_and_send_dns_message = Mock(return_value=(1, 2))
|
|
||||||
|
|
||||||
out = self.agent.delete_zone('ctx', RoObject(name='zn'))
|
|
||||||
|
|
||||||
self.agent._make_and_send_dns_message.assert_called_with(
|
|
||||||
'zn', 1, 14, pcodes.DELETE, pcodes.SUCCESS, 2, 3)
|
|
||||||
self.assertIsNone(out)
|
|
||||||
|
|
||||||
def test_delete_zone_exception(self, *mock):
|
|
||||||
self.agent._make_and_send_dns_message = Mock(return_value=(None, 2))
|
|
||||||
|
|
||||||
with testtools.ExpectedException(exceptions.Backend):
|
|
||||||
self.agent.delete_zone('ctx', RoObject(name='zn'))
|
|
||||||
|
|
||||||
self.agent._make_and_send_dns_message.assert_called_with(
|
|
||||||
'zn', 1, 14, pcodes.DELETE, pcodes.SUCCESS, 2, 3)
|
|
||||||
|
|
||||||
def test_make_and_send_dns_message_timeout(self, *mocks):
|
|
||||||
self.agent._make_dns_message = Mock(return_value='')
|
|
||||||
self.agent._send_dns_message = Mock(
|
|
||||||
return_value=dns.exception.Timeout())
|
|
||||||
|
|
||||||
out = self.agent._make_and_send_dns_message('h', 123, 1, 2, 3, 4, 5)
|
|
||||||
|
|
||||||
self.assertEqual((None, 0), out)
|
|
||||||
|
|
||||||
def test_make_and_send_dns_message_bad_response(self, *mocks):
|
|
||||||
self.agent._make_dns_message = Mock(return_value='')
|
|
||||||
self.agent._send_dns_message = Mock(
|
|
||||||
return_value=agent.dns_query.BadResponse())
|
|
||||||
|
|
||||||
out = self.agent._make_and_send_dns_message('h', 123, 1, 2, 3, 4, 5)
|
|
||||||
|
|
||||||
self.assertEqual((None, 0), out)
|
|
||||||
|
|
||||||
def test_make_and_send_dns_message_missing_AA_flags(self, *mocks):
|
|
||||||
self.agent._make_dns_message = Mock(return_value='')
|
|
||||||
response = RoObject(
|
|
||||||
rcode=Mock(return_value=dns.rcode.NOERROR),
|
|
||||||
# rcode is NOERROR but (flags & dns.flags.AA) gives 0
|
|
||||||
flags=0,
|
|
||||||
)
|
|
||||||
self.agent._send_dns_message = Mock(return_value=response)
|
|
||||||
|
|
||||||
out = self.agent._make_and_send_dns_message('h', 123, 1, 2, 3, 4, 5)
|
|
||||||
|
|
||||||
self.assertEqual((None, 0), out)
|
|
||||||
|
|
||||||
def test_make_and_send_dns_message_error_flags(self, *mocks):
|
|
||||||
self.agent._make_dns_message = Mock(return_value='')
|
|
||||||
response = RoObject(
|
|
||||||
rcode=Mock(return_value=dns.rcode.NOERROR),
|
|
||||||
# rcode is NOERROR but flags are not NOERROR
|
|
||||||
flags=123,
|
|
||||||
ednsflags=321
|
|
||||||
)
|
|
||||||
self.agent._send_dns_message = Mock(return_value=response)
|
|
||||||
|
|
||||||
out = self.agent._make_and_send_dns_message('h', 123, 1, 2, 3, 4, 5)
|
|
||||||
|
|
||||||
self.assertEqual((None, 0), out)
|
|
||||||
|
|
||||||
def test_make_and_send_dns_message(self, *mock):
|
|
||||||
self.agent._make_dns_message = Mock(return_value='')
|
|
||||||
response = RoObject(
|
|
||||||
rcode=Mock(return_value=dns.rcode.NOERROR),
|
|
||||||
flags=agent.dns.flags.AA,
|
|
||||||
ednsflags=321
|
|
||||||
)
|
|
||||||
self.agent._send_dns_message = Mock(return_value=response)
|
|
||||||
|
|
||||||
out = self.agent._make_and_send_dns_message('h', 123, 1, 2, 3, 4, 5)
|
|
||||||
|
|
||||||
self.assertEqual((response, 0), out)
|
|
||||||
|
|
||||||
@mock.patch.object(agent.dns_query, 'tcp')
|
|
||||||
@mock.patch.object(agent.dns_query, 'udp')
|
|
||||||
def test_send_dns_message(self, *mocks):
|
|
||||||
mocks[0].return_value = 'mock udp resp'
|
|
||||||
|
|
||||||
out = self.agent._send_dns_message('msg', 'host', 123, 1)
|
|
||||||
|
|
||||||
assert not agent.dns_query.tcp.called
|
|
||||||
agent.dns_query.udp.assert_called_with('msg', 'host', port=123,
|
|
||||||
timeout=1)
|
|
||||||
self.assertEqual('mock udp resp', out)
|
|
||||||
|
|
||||||
@mock.patch.object(agent.dns_query, 'tcp')
|
|
||||||
@mock.patch.object(agent.dns_query, 'udp')
|
|
||||||
def test_send_dns_message_timeout(self, *mocks):
|
|
||||||
mocks[0].side_effect = dns.exception.Timeout
|
|
||||||
|
|
||||||
out = self.agent._send_dns_message('msg', 'host', 123, 1)
|
|
||||||
|
|
||||||
agent.dns_query.udp.assert_called_with('msg', 'host', port=123,
|
|
||||||
timeout=1)
|
|
||||||
assert isinstance(out, dns.exception.Timeout)
|
|
||||||
|
|
||||||
@mock.patch.object(agent.dns_query, 'tcp')
|
|
||||||
@mock.patch.object(agent.dns_query, 'udp')
|
|
||||||
def test_send_dns_message_bad_response(self, *mocks):
|
|
||||||
mocks[0].side_effect = agent.dns_query.BadResponse
|
|
||||||
|
|
||||||
out = self.agent._send_dns_message('msg', 'host', 123, 1)
|
|
||||||
|
|
||||||
agent.dns_query.udp.assert_called_with('msg', 'host', port=123,
|
|
||||||
timeout=1)
|
|
||||||
assert isinstance(out, agent.dns_query.BadResponse)
|
|
||||||
|
|
||||||
@mock.patch.object(agent.dns_query, 'tcp')
|
|
||||||
@mock.patch.object(agent.dns_query, 'udp')
|
|
||||||
def test_send_dns_message_tcp(self, *mocks):
|
|
||||||
agent.CONF = RoObject({
|
|
||||||
'service:mdns': RoObject(all_tcp=True)
|
|
||||||
})
|
|
||||||
mocks[1].return_value = 'mock tcp resp'
|
|
||||||
|
|
||||||
out = self.agent._send_dns_message('msg', 'host', 123, 1)
|
|
||||||
|
|
||||||
assert not agent.dns_query.udp.called
|
|
||||||
agent.dns_query.tcp.assert_called_with('msg', 'host', port=123,
|
|
||||||
timeout=1)
|
|
||||||
self.assertEqual('mock tcp resp', out)
|
|
@ -1,123 +0,0 @@
|
|||||||
# Copyright 2015 Hewlett-Packard Development Company, L.P.
|
|
||||||
#
|
|
||||||
# Author: Endre Karlson <endre.karlson@hpe.com>
|
|
||||||
#
|
|
||||||
# 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 test Backend
|
|
||||||
"""
|
|
||||||
from designateclient import exceptions
|
|
||||||
from mock import patch
|
|
||||||
from mock import NonCallableMagicMock
|
|
||||||
from mock import Mock
|
|
||||||
from oslo_log import log as logging
|
|
||||||
import fixtures
|
|
||||||
import oslotest.base
|
|
||||||
import testtools
|
|
||||||
|
|
||||||
from designate.utils import generate_uuid
|
|
||||||
from designate import objects
|
|
||||||
from designate.backend import impl_designate
|
|
||||||
|
|
||||||
LOG = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
def create_zone():
|
|
||||||
id_ = generate_uuid()
|
|
||||||
return objects.Zone(
|
|
||||||
id=id_,
|
|
||||||
name='%s-example.com.' % id_,
|
|
||||||
email='root@example.com',
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class RoObject(dict):
|
|
||||||
def __setitem__(self, *a):
|
|
||||||
raise NotImplementedError
|
|
||||||
|
|
||||||
def __setattr__(self, *a):
|
|
||||||
raise NotImplementedError
|
|
||||||
|
|
||||||
def __getattr__(self, k):
|
|
||||||
return self[k]
|
|
||||||
|
|
||||||
|
|
||||||
class DesignateBackendTest(oslotest.base.BaseTestCase):
|
|
||||||
def setUp(self):
|
|
||||||
super(DesignateBackendTest, self).setUp()
|
|
||||||
opts = RoObject(
|
|
||||||
username='user',
|
|
||||||
password='secret',
|
|
||||||
project_name='project',
|
|
||||||
project_zone_name='project_zone',
|
|
||||||
user_zone_name='user_zone'
|
|
||||||
)
|
|
||||||
self.target = RoObject({
|
|
||||||
'id': '4588652b-50e7-46b9-b688-a9bad40a873e',
|
|
||||||
'type': 'dyndns',
|
|
||||||
'masters': [RoObject({'host': '192.0.2.1', 'port': 53})],
|
|
||||||
'options': opts
|
|
||||||
})
|
|
||||||
|
|
||||||
# Backends blow up when trying to self.admin_context = ... due to
|
|
||||||
# policy not being initialized
|
|
||||||
self.admin_context = Mock()
|
|
||||||
self.useFixture(fixtures.MockPatch(
|
|
||||||
'designate.context.DesignateContext.get_admin_context',
|
|
||||||
return_value=self.admin_context
|
|
||||||
))
|
|
||||||
|
|
||||||
self.backend = impl_designate.DesignateBackend(self.target)
|
|
||||||
|
|
||||||
# Mock client
|
|
||||||
self.client = NonCallableMagicMock()
|
|
||||||
zones = NonCallableMagicMock(spec_set=[
|
|
||||||
'create', 'delete'])
|
|
||||||
self.client.configure_mock(zones=zones)
|
|
||||||
|
|
||||||
def test_create_zone(self):
|
|
||||||
zone = create_zone()
|
|
||||||
masters = ["%(host)s:%(port)s" % self.target.masters[0]]
|
|
||||||
with patch.object(
|
|
||||||
self.backend, '_get_client', return_value=self.client):
|
|
||||||
self.backend.create_zone(self.admin_context, zone)
|
|
||||||
self.client.zones.create.assert_called_once_with(
|
|
||||||
zone.name, 'SECONDARY', masters=masters)
|
|
||||||
|
|
||||||
def test_delete_zone(self):
|
|
||||||
zone = create_zone()
|
|
||||||
with patch.object(
|
|
||||||
self.backend, '_get_client', return_value=self.client):
|
|
||||||
self.backend.delete_zone(self.admin_context, zone)
|
|
||||||
self.client.zones.delete.assert_called_once_with(zone.name)
|
|
||||||
|
|
||||||
def test_delete_zone_notfound(self):
|
|
||||||
zone = create_zone()
|
|
||||||
self.client.delete.side_effect = exceptions.NotFound
|
|
||||||
with patch.object(
|
|
||||||
self.backend, '_get_client', return_value=self.client):
|
|
||||||
self.backend.delete_zone(self.admin_context, zone)
|
|
||||||
self.client.zones.delete.assert_called_once_with(zone.name)
|
|
||||||
|
|
||||||
def test_delete_zone_exc(self):
|
|
||||||
class Exc(Exception):
|
|
||||||
pass
|
|
||||||
|
|
||||||
zone = create_zone()
|
|
||||||
self.client.zones.delete.side_effect = Exc()
|
|
||||||
with testtools.ExpectedException(Exc):
|
|
||||||
with patch.object(
|
|
||||||
self.backend, '_get_client', return_value=self.client):
|
|
||||||
self.backend.delete_zone(self.admin_context, zone)
|
|
||||||
self.client.zones.delete.assert_called_once_with(zone.name)
|
|
Loading…
Reference in New Issue
Block a user