![Drew Freiberger](/assets/img/avatar_default.png)
Designate-bind services use http communication to manage initial zone transfers from the leader to other units. This should be done within the space designated for dns-backend in the charm bindings. To ensure this, we must bypass proxy configurations when using wget between units by adding --no-proxy flag to the wget command. Change-Id: I3cebb1e01ffde9a9585f152451bf9bcebbdd3f58 Closes-Bug: #1796969
458 lines
18 KiB
Python
458 lines
18 KiB
Python
# Copyright 2016 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 unittest
|
|
|
|
|
|
import charm.openstack.designate_bind as designate_bind
|
|
|
|
|
|
def FakeConfig(init_dict):
|
|
|
|
def _config(key=None):
|
|
return init_dict[key] if key else init_dict
|
|
|
|
return _config
|
|
|
|
|
|
class Helper(unittest.TestCase):
|
|
|
|
def setUp(self):
|
|
self._patches = {}
|
|
self._patches_start = {}
|
|
self.ch_config_patch = mock.patch('charmhelpers.core.hookenv.config')
|
|
self.ch_config = self.ch_config_patch.start()
|
|
self.ch_config.side_effect = lambda: {'ssl_param': None}
|
|
|
|
def tearDown(self):
|
|
for k, v in self._patches.items():
|
|
v.stop()
|
|
setattr(self, k, None)
|
|
self._patches = None
|
|
self._patches_start = None
|
|
self.ch_config_patch.stop()
|
|
|
|
def patch(self, obj, attr, return_value=None, **kwargs):
|
|
mocked = mock.patch.object(obj, attr, **kwargs)
|
|
self._patches[attr] = mocked
|
|
started = mocked.start()
|
|
started.return_value = return_value
|
|
self._patches_start[attr] = started
|
|
setattr(self, attr, started)
|
|
|
|
def patch_object(self, obj, attr, return_value=None, name=None, new=None):
|
|
if name is None:
|
|
name = attr
|
|
if new is not None:
|
|
mocked = mock.patch.object(obj, attr, new=new)
|
|
else:
|
|
mocked = mock.patch.object(obj, attr)
|
|
self._patches[name] = mocked
|
|
started = mocked.start()
|
|
if new is None:
|
|
started.return_value = return_value
|
|
self._patches_start[name] = started
|
|
setattr(self, name, started)
|
|
|
|
|
|
class TestOpenStackDesignateBind(Helper):
|
|
|
|
def test_install(self):
|
|
self.patch(designate_bind.DesignateBindCharm.singleton, 'install')
|
|
designate_bind.install()
|
|
self.install.assert_called_once_with()
|
|
|
|
def test_init_rndckey(self):
|
|
self.patch(designate_bind.DesignateBindCharm.singleton, 'init_rndckey')
|
|
designate_bind.init_rndckey()
|
|
self.init_rndckey.assert_called_once_with()
|
|
|
|
def test_get_rndc_secret(self):
|
|
self.patch(
|
|
designate_bind.DesignateBindCharm.singleton,
|
|
'get_rndc_secret')
|
|
designate_bind.get_rndc_secret()
|
|
self.get_rndc_secret.assert_called_once_with()
|
|
|
|
def test_get_rndc_algorithm(self):
|
|
self.patch(
|
|
designate_bind.DesignateBindCharm.singleton,
|
|
'get_rndc_algorithm')
|
|
designate_bind.get_rndc_algorithm()
|
|
self.get_rndc_algorithm.assert_called_once_with()
|
|
|
|
def test_get_sync_time(self):
|
|
self.patch(
|
|
designate_bind.DesignateBindCharm.singleton,
|
|
'get_sync_time')
|
|
designate_bind.get_sync_time()
|
|
self.get_sync_time.assert_called_once_with()
|
|
|
|
def test_setup_sync(self):
|
|
self.patch(designate_bind.DesignateBindCharm.singleton, 'setup_sync')
|
|
designate_bind.setup_sync()
|
|
self.setup_sync.assert_called_once_with()
|
|
|
|
def test_retrieve_zones(self):
|
|
self.patch(
|
|
designate_bind.DesignateBindCharm.singleton,
|
|
'retrieve_zones')
|
|
designate_bind.retrieve_zones('hacluster')
|
|
self.retrieve_zones.assert_called_once_with('hacluster')
|
|
|
|
def test_request_sync(self):
|
|
self.patch(
|
|
designate_bind.DesignateBindCharm.singleton,
|
|
'request_sync')
|
|
designate_bind.request_sync('hacluster')
|
|
self.request_sync.assert_called_once_with('hacluster')
|
|
|
|
def test_process_requests(self):
|
|
self.patch(
|
|
designate_bind.DesignateBindCharm.singleton,
|
|
'process_requests')
|
|
designate_bind.process_requests('hacluster')
|
|
self.process_requests.assert_called_once_with('hacluster')
|
|
|
|
def test_render_all_configs(self):
|
|
self.patch(
|
|
designate_bind.DesignateBindCharm.singleton,
|
|
'render_with_interfaces')
|
|
designate_bind.render_all_configs('interface_list')
|
|
self.render_with_interfaces.assert_called_once_with('interface_list')
|
|
|
|
|
|
class TestDNSAdapter(Helper):
|
|
|
|
def test_control_listen_ip(self):
|
|
relation = mock.MagicMock()
|
|
self.patch(designate_bind.ch_ip, 'get_relation_ip')
|
|
self.patch(designate_bind.hookenv, 'unit_private_ip')
|
|
self.get_relation_ip.return_value = 'ip1'
|
|
a = designate_bind.DNSAdapter(relation)
|
|
self.assertEqual(a.control_listen_ip, 'ip1')
|
|
|
|
def test_control_ips(self):
|
|
relation = mock.MagicMock()
|
|
relation.client_ips.return_value = ['ip1', 'ip2']
|
|
a = designate_bind.DNSAdapter(relation)
|
|
self.assertEqual(a.control_ips, 'ip1;ip2')
|
|
|
|
def test_algorithm(self):
|
|
relation = mock.MagicMock()
|
|
self.patch(designate_bind.DesignateBindCharm, 'get_rndc_algorithm')
|
|
self.get_rndc_algorithm.return_value = 'algo1'
|
|
a = designate_bind.DNSAdapter(relation)
|
|
self.assertEqual(a.algorithm, 'algo1')
|
|
|
|
def test_secret(self):
|
|
relation = mock.MagicMock()
|
|
self.patch(designate_bind.DesignateBindCharm, 'get_rndc_secret')
|
|
self.get_rndc_secret.return_value = 'secret1'
|
|
a = designate_bind.DNSAdapter(relation)
|
|
self.assertEqual(a.secret, 'secret1')
|
|
|
|
|
|
class TestBindAdapters(Helper):
|
|
|
|
def test_bind_adapters(self):
|
|
dns_backend_relation = mock.MagicMock()
|
|
dns_backend_relation.endpoint_name = 'dns_backend'
|
|
b = designate_bind.BindAdapters([dns_backend_relation])
|
|
# ensure that the relevant things got put on.
|
|
self.assertTrue(
|
|
isinstance(
|
|
b.dns_backend,
|
|
designate_bind.adapters.OpenStackRelationAdapter))
|
|
|
|
|
|
class TestDesignateBindCharm(Helper):
|
|
|
|
def test_get_rndc_algorithm(self):
|
|
self.assertEqual(
|
|
designate_bind.DesignateBindCharm.get_rndc_algorithm(),
|
|
'hmac-md5'
|
|
)
|
|
|
|
def test_get_rndc_secret(self):
|
|
self.patch(designate_bind.hookenv, 'leader_get')
|
|
self.leader_get.return_value = 'secret1'
|
|
self.assertEqual(
|
|
designate_bind.DesignateBindCharm.get_rndc_secret(),
|
|
'secret1'
|
|
)
|
|
|
|
def test_get_sync_src(self):
|
|
self.patch(designate_bind.hookenv, 'leader_get')
|
|
self.leader_get.return_value = 'http://ip1/my.tar'
|
|
self.assertEqual(
|
|
designate_bind.DesignateBindCharm.get_sync_src(),
|
|
'http://ip1/my.tar'
|
|
)
|
|
|
|
def test_get_sync_time(self):
|
|
self.patch(designate_bind.hookenv, 'leader_get')
|
|
self.leader_get.return_value = '100'
|
|
self.assertEqual(
|
|
designate_bind.DesignateBindCharm.get_sync_time(),
|
|
'100'
|
|
)
|
|
|
|
def test_process_requests(self):
|
|
hacluster = mock.MagicMock()
|
|
self.patch(designate_bind.hookenv, 'log')
|
|
self.patch(designate_bind.DesignateBindCharm, 'setup_sync')
|
|
self.patch(designate_bind.DesignateBindCharm, 'get_sync_time')
|
|
a = designate_bind.DesignateBindCharm()
|
|
# No queued requests
|
|
hacluster.retrieve_remote.return_value = []
|
|
self.get_sync_time.return_value = 20
|
|
a.process_requests(hacluster)
|
|
self.assertFalse(self.setup_sync.called)
|
|
# No request since last sync
|
|
self.setup_sync.reset_mock()
|
|
hacluster.retrieve_remote.return_value = ['10']
|
|
self.get_sync_time.return_value = 20
|
|
a.process_requests(hacluster)
|
|
self.assertFalse(self.setup_sync.called)
|
|
# New request present
|
|
self.setup_sync.reset_mock()
|
|
hacluster.retrieve_remote.return_value = ['10', '30']
|
|
self.get_sync_time.return_value = 20
|
|
a.process_requests(hacluster)
|
|
self.assertTrue(self.setup_sync.called)
|
|
|
|
def test_set_sync_info(self):
|
|
self.patch(designate_bind.hookenv, 'leader_set')
|
|
self.patch(designate_bind.hookenv, 'unit_private_ip')
|
|
self.unit_private_ip.return_value = 'ip1'
|
|
a = designate_bind.DesignateBindCharm()
|
|
a.set_sync_info('20', '/tmp/tarball.tar')
|
|
self.leader_set.assert_called_once_with({
|
|
'sync_time': '20',
|
|
'sync_src': 'http://ip1:80/zone-syncs//tmp/tarball.tar'})
|
|
|
|
def test_generate_rndc_key(self):
|
|
hmac_mock = mock.MagicMock()
|
|
self.patch(designate_bind.os, 'urandom', return_value='seed')
|
|
self.patch(designate_bind.hmac, 'new', return_value=hmac_mock)
|
|
self.patch(designate_bind.base64, 'b64encode', return_value=hmac_mock)
|
|
self.patch_object(designate_bind.hashlib, 'md5', new='md5lib')
|
|
a = designate_bind.DesignateBindCharm()
|
|
a.generate_rndc_key()
|
|
self.new.assert_called_once_with(
|
|
'seed',
|
|
digestmod='md5lib',
|
|
msg=b'RNDC Secret')
|
|
|
|
def test_init_rndckey(self):
|
|
self.patch(designate_bind.hookenv, 'log')
|
|
self.patch(designate_bind.DesignateBindCharm, 'get_rndc_secret')
|
|
self.patch(designate_bind.DesignateBindCharm, 'generate_rndc_key')
|
|
self.patch(designate_bind.hookenv, 'leader_set')
|
|
self.patch(designate_bind.hookenv, 'is_leader')
|
|
a = designate_bind.DesignateBindCharm()
|
|
# Test secret already stored
|
|
self.get_rndc_secret.return_value = 'mysecret'
|
|
self.assertEqual(a.init_rndckey(), 'mysecret')
|
|
# Test need new secret (Leader)
|
|
self.get_rndc_secret.return_value = None
|
|
self.generate_rndc_key.return_value = 'newsecret'
|
|
self.is_leader.return_value = True
|
|
self.assertEqual(a.init_rndckey(), 'newsecret')
|
|
self.leader_set.assert_called_once_with({'rndc_key': 'newsecret'})
|
|
# Test need new secret (Not Leader)
|
|
self.get_rndc_secret.return_value = None
|
|
self.is_leader.return_value = False
|
|
self.assertEqual(a.init_rndckey(), None)
|
|
|
|
def test_create_zone_tarball(self):
|
|
self.patch(designate_bind.glob, 'glob')
|
|
self.patch(designate_bind.subprocess, 'check_call')
|
|
_files = {
|
|
'/var/cache/bind/juju*': ['jujufile1'],
|
|
'/var/cache/bind/slave*': ['slavefile1'],
|
|
'/var/cache/bind/*nzf': ['nsffile']}
|
|
self.glob.side_effect = lambda x: _files[x]
|
|
a = designate_bind.DesignateBindCharm()
|
|
a.create_zone_tarball('/tmp/tarball.tar')
|
|
self.check_call.assert_called_once_with([
|
|
'tar', 'zcvf', '/tmp/tarball.tar', 'jujufile1', 'slavefile1',
|
|
'nsffile'], cwd='/var/cache/bind')
|
|
|
|
def test_setup_sync_dir(self):
|
|
self.patch(designate_bind.os, 'mkdir')
|
|
self.patch(designate_bind.os, 'chmod')
|
|
a = designate_bind.DesignateBindCharm()
|
|
a.setup_sync_dir('100')
|
|
self.mkdir.assert_called_once_with('/var/www/html/zone-syncs', 493)
|
|
self.assertFalse(self.chmod.called)
|
|
# Test dir does not exist
|
|
self.mkdir.side_effect = FileExistsError
|
|
a.setup_sync_dir('100')
|
|
self.chmod.assert_called_once_with('/var/www/html/zone-syncs', 493)
|
|
|
|
def test_create_sync_src_info_file(self):
|
|
self.patch(designate_bind.hookenv, 'local_unit', return_value='unit/1')
|
|
a = designate_bind.DesignateBindCharm()
|
|
with mock.patch('builtins.open') as bob:
|
|
a.create_sync_src_info_file()
|
|
bob.assert_called_once_with(
|
|
'/var/cache/bind/juju-zone-src-unit_1',
|
|
'w+')
|
|
|
|
def test_setup_sync(self):
|
|
self.patch(designate_bind.hookenv, 'log')
|
|
self.patch(designate_bind.DesignateBindCharm, 'setup_sync_dir')
|
|
self.patch(designate_bind.time, 'time')
|
|
self.patch(
|
|
designate_bind.DesignateBindCharm,
|
|
'create_sync_src_info_file')
|
|
self.patch(designate_bind.DesignateBindCharm, 'service_control')
|
|
self.patch(designate_bind.DesignateBindCharm, 'create_zone_tarball')
|
|
self.patch(designate_bind.DesignateBindCharm, 'set_sync_info')
|
|
self.setup_sync_dir.return_value = '/tmp/zonefiles'
|
|
self.time.return_value = 100
|
|
a = designate_bind.DesignateBindCharm()
|
|
a.setup_sync()
|
|
self.setup_sync_dir.assert_called_once_with('100')
|
|
self.create_sync_src_info_file.assert_called_once_with()
|
|
ctrl_calls = [
|
|
mock.call('stop', ['bind9']),
|
|
mock.call('start', ['bind9'])]
|
|
self.service_control.assert_has_calls(ctrl_calls)
|
|
self.create_zone_tarball.assert_called_once_with(
|
|
'/tmp/zonefiles/100.tar.gz')
|
|
self.set_sync_info.assert_called_once_with('100', '100.tar.gz')
|
|
|
|
def test_service_control(self):
|
|
self.patch(designate_bind.host, 'service_stop')
|
|
a = designate_bind.DesignateBindCharm()
|
|
a.service_control('stop', ['svc1', 'svc2'])
|
|
ctrl_calls = [
|
|
mock.call('svc1'),
|
|
mock.call('svc2')]
|
|
self.service_stop.assert_has_calls(ctrl_calls)
|
|
|
|
def test_request_sync(self):
|
|
self.patch(designate_bind.time, 'time')
|
|
relation = mock.MagicMock()
|
|
self.patch(designate_bind.reactive, 'set_state')
|
|
self.time.return_value = 100
|
|
a = designate_bind.DesignateBindCharm()
|
|
a.request_sync(relation)
|
|
relation.send_all.assert_called_once_with(
|
|
{'sync_request': '100'},
|
|
store_local=True)
|
|
self.set_state.assert_called_once_with('sync.request.sent')
|
|
|
|
def test_wget_file(self):
|
|
# retry_on_exception patched out in __init__.py
|
|
self.patch(designate_bind.subprocess, 'check_call')
|
|
a = designate_bind.DesignateBindCharm()
|
|
a.wget_file('http://ip1/tarfile.tar', '/tmp')
|
|
self.check_call.assert_called_once_with(
|
|
['wget', 'http://ip1/tarfile.tar', '--no-proxy',
|
|
'--retry-connrefused', '-t', '10'],
|
|
cwd='/tmp'
|
|
)
|
|
|
|
def test_retrieve_zones_cluster_relation(self):
|
|
relation = mock.MagicMock()
|
|
self.patch(designate_bind.DesignateBindCharm, 'get_sync_time')
|
|
self.patch(designate_bind.DesignateBindCharm, 'get_sync_src')
|
|
self.patch(designate_bind.DesignateBindCharm, 'service_control')
|
|
self.patch(designate_bind.hookenv, 'log')
|
|
self.patch(designate_bind.reactive, 'set_state')
|
|
self.patch(designate_bind.reactive, 'remove_state')
|
|
self.patch(designate_bind.os, 'remove')
|
|
self.patch(designate_bind.subprocess, 'check_call')
|
|
self.patch_object(designate_bind.DesignateBindCharm, 'wget_file')
|
|
self.get_sync_src.return_value = 'http://ip1/tarfile.tar'
|
|
ctrl_calls = [
|
|
mock.call('stop', ['bind9']),
|
|
mock.call('start', ['bind9'])]
|
|
a = designate_bind.DesignateBindCharm()
|
|
# Using cluster_relation, no sync needed
|
|
relation.retrieve_local.return_value = ['30']
|
|
self.get_sync_time.return_value = '20'
|
|
a.retrieve_zones(relation)
|
|
self.assertFalse(self.service_control.called)
|
|
# Using cluster_relation, sync needed
|
|
self.service_control.reset_mock()
|
|
relation.retrieve_local.return_value = ['10']
|
|
self.get_sync_time.return_value = '20'
|
|
a.retrieve_zones(relation)
|
|
self.service_control.assert_has_calls(ctrl_calls)
|
|
self.check_call.assert_called_once_with(
|
|
['tar', 'xf', 'tarfile.tar'], cwd='/var/cache/bind')
|
|
self.wget_file.assert_called_once_with(
|
|
'http://ip1/tarfile.tar',
|
|
'/var/cache/bind')
|
|
|
|
def test_retrieve_zones_cluster_relation_nourl(self):
|
|
relation = mock.MagicMock()
|
|
self.patch(designate_bind.DesignateBindCharm, 'get_sync_time')
|
|
self.patch(designate_bind.DesignateBindCharm, 'get_sync_src')
|
|
self.patch_object(designate_bind.DesignateBindCharm, 'wget_file')
|
|
self.patch(designate_bind.hookenv, 'log')
|
|
self.get_sync_src.return_value = None
|
|
relation.retrieve_local.return_value = ['10']
|
|
self.get_sync_time.return_value = '20'
|
|
a = designate_bind.DesignateBindCharm()
|
|
a.retrieve_zones(relation)
|
|
self.assertFalse(self.wget_file.called)
|
|
|
|
def test_retrieve_zones_no_cluster_relation(self):
|
|
self.patch(designate_bind.DesignateBindCharm, 'get_sync_time')
|
|
self.patch(designate_bind.DesignateBindCharm, 'get_sync_src')
|
|
self.patch(designate_bind.DesignateBindCharm, 'service_control')
|
|
self.patch(designate_bind.hookenv, 'log')
|
|
self.patch(designate_bind.reactive, 'set_state')
|
|
self.patch(designate_bind.reactive, 'remove_state')
|
|
self.patch(designate_bind.os, 'remove')
|
|
self.patch(designate_bind.subprocess, 'check_call')
|
|
self.patch_object(designate_bind.DesignateBindCharm, 'wget_file')
|
|
self.get_sync_src.return_value = 'http://ip1/tarfile.tar'
|
|
ctrl_calls = [
|
|
mock.call('stop', ['bind9']),
|
|
mock.call('start', ['bind9'])]
|
|
a = designate_bind.DesignateBindCharm()
|
|
self.get_sync_time.return_value = '20'
|
|
a.retrieve_zones()
|
|
self.service_control.assert_has_calls(ctrl_calls)
|
|
self.check_call.assert_called_once_with(
|
|
['tar', 'xf', 'tarfile.tar'], cwd='/var/cache/bind')
|
|
self.wget_file.assert_called_once_with(
|
|
'http://ip1/tarfile.tar',
|
|
'/var/cache/bind')
|
|
|
|
def test_set_apparmor(self):
|
|
self.patch(designate_bind.os.path, 'isfile')
|
|
a = designate_bind.DesignateBindCharm()
|
|
self.isfile.return_value = True
|
|
with mock.patch('builtins.open') as bob:
|
|
a.set_apparmor()
|
|
self.assertFalse(bob.called)
|
|
self.isfile.return_value = False
|
|
with mock.patch('builtins.open') as bob:
|
|
a.set_apparmor()
|
|
bob.assert_called_once_with(
|
|
'/etc/apparmor.d/disable/usr.sbin.named',
|
|
'w')
|