Merge "Add NSD4 backend"
This commit is contained in:
commit
452ef4eb3c
@ -1,140 +0,0 @@
|
|||||||
# Copyright (C) 2013 eNovance SAS <licensing@enovance.com>
|
|
||||||
#
|
|
||||||
# Author: Artom Lifshitz <artom.lifshitz@enovance.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 os
|
|
||||||
import socket
|
|
||||||
import ssl
|
|
||||||
|
|
||||||
import eventlet
|
|
||||||
from oslo_config import cfg
|
|
||||||
from oslo_log import log as logging
|
|
||||||
|
|
||||||
from designate import exceptions
|
|
||||||
from designate import utils
|
|
||||||
from designate.backend import base
|
|
||||||
|
|
||||||
|
|
||||||
LOG = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
CFG_GROUP = 'backend:nsd4slave'
|
|
||||||
DEFAULT_PORT = 8952
|
|
||||||
|
|
||||||
|
|
||||||
class NSD4SlaveBackend(base.Backend):
|
|
||||||
__plugin__name__ = 'nsd4slave'
|
|
||||||
NSDCT_VERSION = 'NSDCT1'
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def get_cfg_opts(cls):
|
|
||||||
group = cfg.OptGroup(
|
|
||||||
name=CFG_GROUP, title="Configuration for NSD4-slave backend"
|
|
||||||
)
|
|
||||||
|
|
||||||
opts = [
|
|
||||||
cfg.StrOpt('keyfile', default='/etc/nsd/nsd_control.key',
|
|
||||||
help='Keyfile to use when connecting to the NSD4 '
|
|
||||||
'servers over SSL'),
|
|
||||||
cfg.StrOpt('certfile', default='/etc/nsd/nsd_control.pem',
|
|
||||||
help='Certfile to use when connecting to the NSD4 '
|
|
||||||
'servers over SSL'),
|
|
||||||
cfg.ListOpt('servers',
|
|
||||||
help='Comma-separated list of servers to control, in '
|
|
||||||
' <host>:<port> format. If <port> is omitted, '
|
|
||||||
' the default 8952 is used.'),
|
|
||||||
cfg.StrOpt('pattern', default='slave',
|
|
||||||
help='Pattern to use when creating zones on the NSD4 '
|
|
||||||
'servers. This pattern must be identically '
|
|
||||||
'configured on all NSD4 servers.'),
|
|
||||||
]
|
|
||||||
|
|
||||||
return [(group, opts)]
|
|
||||||
|
|
||||||
def __init__(self, central_service):
|
|
||||||
self._keyfile = cfg.CONF[CFG_GROUP].keyfile
|
|
||||||
self._certfile = cfg.CONF[CFG_GROUP].certfile
|
|
||||||
# Make sure keyfile and certfile are readable to avoid cryptic SSL
|
|
||||||
# errors later
|
|
||||||
if not os.access(self._keyfile, os.R_OK):
|
|
||||||
raise exceptions.NSD4SlaveBackendError(
|
|
||||||
'Keyfile %s missing or permission denied' % self._keyfile)
|
|
||||||
if not os.access(self._certfile, os.R_OK):
|
|
||||||
raise exceptions.NSD4SlaveBackendError(
|
|
||||||
'Certfile %s missing or permission denied' % self._certfile)
|
|
||||||
self._pattern = cfg.CONF[CFG_GROUP].pattern
|
|
||||||
try:
|
|
||||||
self._servers = [self._parse_server(cfg_server)
|
|
||||||
for cfg_server in cfg.CONF[CFG_GROUP].servers]
|
|
||||||
except TypeError:
|
|
||||||
raise exceptions.ConfigurationError('No NSD4 servers defined')
|
|
||||||
|
|
||||||
def _parse_server(self, cfg_server):
|
|
||||||
try:
|
|
||||||
(host, port) = utils.split_host_port(cfg_server)
|
|
||||||
port = int(port)
|
|
||||||
except ValueError:
|
|
||||||
host = str(cfg_server)
|
|
||||||
port = DEFAULT_PORT
|
|
||||||
return {'host': host, 'port': port}
|
|
||||||
|
|
||||||
def create_domain(self, context, domain):
|
|
||||||
command = 'addzone %s %s' % (domain['name'], self._pattern)
|
|
||||||
self._all_servers(command)
|
|
||||||
|
|
||||||
def update_domain(self, context, domain):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def delete_domain(self, context, domain):
|
|
||||||
command = 'delzone %s' % domain['name']
|
|
||||||
self._all_servers(command)
|
|
||||||
|
|
||||||
def _all_servers(self, command):
|
|
||||||
for server in self._servers:
|
|
||||||
try:
|
|
||||||
result = self._command(command, server['host'], server['port'])
|
|
||||||
except (ssl.SSLError, socket.error) as e:
|
|
||||||
raise exceptions.NSD4SlaveBackendError(e)
|
|
||||||
if result != 'ok':
|
|
||||||
raise exceptions.NSD4SlaveBackendError(result)
|
|
||||||
|
|
||||||
def _command(self, command, host, port):
|
|
||||||
sock = eventlet.wrap_ssl(eventlet.connect((host, port)),
|
|
||||||
keyfile=self._keyfile,
|
|
||||||
certfile=self._certfile)
|
|
||||||
stream = sock.makefile()
|
|
||||||
stream.write('%s %s\n' % (self.NSDCT_VERSION, command))
|
|
||||||
stream.flush()
|
|
||||||
result = stream.read()
|
|
||||||
stream.close()
|
|
||||||
sock.close()
|
|
||||||
return result.rstrip()
|
|
||||||
|
|
||||||
def create_recordset(self, context, domain, recordset):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def update_recordset(self, context, domain, recordset):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def delete_recordset(self, context, domain, recordset):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def create_record(self, context, domain, recordset, record):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def update_record(self, context, domain, recordset, record):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def delete_record(self, context, domain, recordset, record):
|
|
||||||
pass
|
|
103
designate/backend/impl_nsd4.py
Normal file
103
designate/backend/impl_nsd4.py
Normal file
@ -0,0 +1,103 @@
|
|||||||
|
# Copyright (C) 2013 eNovance SAS <licensing@enovance.com>
|
||||||
|
# Copyright 2014 eBay Inc.
|
||||||
|
# Copyright 2015 Zetta.IO.
|
||||||
|
#
|
||||||
|
# Author: Ron Rickard <rrickard@ebay.com>
|
||||||
|
# Author: Artom Lifshitz <artom.lifshitz@enovance.com>
|
||||||
|
# Author: Dag Stenstad <dag@stenstad.net>
|
||||||
|
#
|
||||||
|
# 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 random
|
||||||
|
import socket
|
||||||
|
import ssl
|
||||||
|
|
||||||
|
import eventlet
|
||||||
|
from oslo_log import log as logging
|
||||||
|
|
||||||
|
from designate import exceptions
|
||||||
|
from designate.backend import base
|
||||||
|
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class NSD4Backend(base.Backend):
|
||||||
|
__plugin_name__ = 'nsd4'
|
||||||
|
NSDCT_VERSION = 'NSDCT1'
|
||||||
|
|
||||||
|
def __init__(self, target):
|
||||||
|
super(NSD4Backend, self).__init__(target)
|
||||||
|
|
||||||
|
self.host = self.options.get('host', '127.0.0.1')
|
||||||
|
self.port = int(self.options.get('port', 8952))
|
||||||
|
self.certfile = self.options.get('certfile',
|
||||||
|
'/etc/nsd/nsd_control.pem')
|
||||||
|
self.keyfile = self.options.get('keyfile',
|
||||||
|
'/etc/nsd/nsd_control.key')
|
||||||
|
self.pattern = self.options.get('pattern', 'slave')
|
||||||
|
|
||||||
|
def _command(self, command):
|
||||||
|
sock = eventlet.wrap_ssl(
|
||||||
|
eventlet.connect((self.host, self.port)),
|
||||||
|
keyfile=self.keyfile,
|
||||||
|
certfile=self.certfile)
|
||||||
|
stream = sock.makefile()
|
||||||
|
stream.write('%s %s\n' % (self.NSDCT_VERSION, command))
|
||||||
|
stream.flush()
|
||||||
|
result = stream.read()
|
||||||
|
stream.close()
|
||||||
|
sock.close()
|
||||||
|
return result
|
||||||
|
|
||||||
|
def _execute_nsd4(self, command):
|
||||||
|
try:
|
||||||
|
LOG.debug('Executing NSD4 control call: %s on %s' % (command,
|
||||||
|
self.host))
|
||||||
|
result = self._command(command)
|
||||||
|
except (ssl.SSLError, socket.error) as e:
|
||||||
|
LOG.debug('NSD4 control call failure: %s' % e)
|
||||||
|
raise exceptions.Backend(e)
|
||||||
|
if result != 'ok':
|
||||||
|
raise exceptions.Backend(result)
|
||||||
|
|
||||||
|
def create_domain(self, context, domain):
|
||||||
|
LOG.debug('Create Domain')
|
||||||
|
masters = []
|
||||||
|
for master in self.masters:
|
||||||
|
host = master['host']
|
||||||
|
port = master['port']
|
||||||
|
masters.append('%s port %s' % (host, port))
|
||||||
|
|
||||||
|
# Ensure different MiniDNS instances are targetted for AXFRs
|
||||||
|
random.shuffle(masters)
|
||||||
|
|
||||||
|
command = 'addzone %s %s' % (domain['name'], self.pattern)
|
||||||
|
|
||||||
|
try:
|
||||||
|
self._execute_nsd4(command)
|
||||||
|
except exceptions.Backend as e:
|
||||||
|
# If create fails because the domain exists, don't reraise
|
||||||
|
if "already exists" not in str(e.message):
|
||||||
|
raise
|
||||||
|
|
||||||
|
def delete_domain(self, context, domain):
|
||||||
|
LOG.debug('Delete Domain')
|
||||||
|
command = 'delzone %s' % domain['name']
|
||||||
|
|
||||||
|
try:
|
||||||
|
self._execute_nsd4(command)
|
||||||
|
except exceptions.Backend as e:
|
||||||
|
# If domain is already deleted, don't reraise
|
||||||
|
if "not found" not in str(e.message):
|
||||||
|
raise
|
138
designate/tests/test_backend/test_nsd4.py
Normal file
138
designate/tests/test_backend/test_nsd4.py
Normal file
@ -0,0 +1,138 @@
|
|||||||
|
# Copyright (C) 2013 eNovance SAS <licensing@enovance.com>
|
||||||
|
#
|
||||||
|
# Author: Artom Lifshitz <artom.lifshitz@enovance.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 os
|
||||||
|
import socket
|
||||||
|
import ssl
|
||||||
|
|
||||||
|
import eventlet
|
||||||
|
import fixtures
|
||||||
|
from mock import MagicMock
|
||||||
|
|
||||||
|
from designate import exceptions
|
||||||
|
from designate import objects
|
||||||
|
from designate.tests.test_backend import BackendTestCase
|
||||||
|
from designate.tests import resources
|
||||||
|
from designate.backend import impl_nsd4
|
||||||
|
|
||||||
|
|
||||||
|
class NSD4ServerStub:
|
||||||
|
recved_command = None
|
||||||
|
response = 'ok'
|
||||||
|
keyfile = os.path.join(resources.path, 'ssl', 'nsd_server.key')
|
||||||
|
certfile = os.path.join(resources.path, 'ssl', 'nsd_server.pem')
|
||||||
|
|
||||||
|
def handle(self, client_sock, client_addr):
|
||||||
|
stream = client_sock.makefile()
|
||||||
|
self.recved_command = stream.readline()
|
||||||
|
stream.write(self.response)
|
||||||
|
stream.flush()
|
||||||
|
|
||||||
|
def start(self):
|
||||||
|
self.port = 1025
|
||||||
|
while True:
|
||||||
|
try:
|
||||||
|
eventlet.spawn_n(eventlet.serve,
|
||||||
|
eventlet.wrap_ssl(
|
||||||
|
eventlet.listen(('127.0.0.1', self.port)),
|
||||||
|
keyfile=self.keyfile,
|
||||||
|
certfile=self.certfile,
|
||||||
|
server_side=True),
|
||||||
|
self.handle)
|
||||||
|
break
|
||||||
|
except socket.error:
|
||||||
|
self.port = self.port + 1
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
eventlet.StopServe()
|
||||||
|
|
||||||
|
|
||||||
|
class NSD4Fixture(fixtures.Fixture):
|
||||||
|
def setUp(self):
|
||||||
|
super(NSD4Fixture, self).setUp()
|
||||||
|
self.server = NSD4ServerStub()
|
||||||
|
self.server.start()
|
||||||
|
|
||||||
|
self.addCleanup(self.tearDown)
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
self.server.stop()
|
||||||
|
|
||||||
|
|
||||||
|
# NOTE: We'll only test the specifics to the nsd4 backend here.
|
||||||
|
# Rest is handled via scenarios
|
||||||
|
class NSD4BackendTestCase(BackendTestCase):
|
||||||
|
def setUp(self):
|
||||||
|
super(NSD4BackendTestCase, self).setUp()
|
||||||
|
|
||||||
|
self.server_fixture = NSD4Fixture()
|
||||||
|
self.useFixture(self.server_fixture)
|
||||||
|
|
||||||
|
keyfile = os.path.join(resources.path, 'ssl', 'nsd_control.key')
|
||||||
|
certfile = os.path.join(resources.path, 'ssl', 'nsd_control.pem')
|
||||||
|
|
||||||
|
self.target = objects.PoolTarget.from_dict({
|
||||||
|
'id': '4588652b-50e7-46b9-b688-a9bad40a873e',
|
||||||
|
'type': 'nsd4',
|
||||||
|
'masters': [{'host': '192.0.2.1', 'port': 53},
|
||||||
|
{'host': '192.0.2.2', 'port': 35}],
|
||||||
|
'options': [
|
||||||
|
{'key': 'keyfile', 'value': keyfile},
|
||||||
|
{'key': 'certfile', 'value': certfile},
|
||||||
|
{'key': 'pattern', 'value': 'test-pattern'},
|
||||||
|
{'key': 'port', 'value': self.server_fixture.server.port}
|
||||||
|
],
|
||||||
|
})
|
||||||
|
|
||||||
|
self.backend = impl_nsd4.NSD4Backend(self.target)
|
||||||
|
|
||||||
|
def test_create_domain(self):
|
||||||
|
context = self.get_context()
|
||||||
|
domain = self.get_domain_fixture()
|
||||||
|
self.backend.create_domain(context, domain)
|
||||||
|
command = 'NSDCT1 addzone %s test-pattern\n' % domain['name']
|
||||||
|
self.assertEqual(command, self.server_fixture.server.recved_command)
|
||||||
|
|
||||||
|
def test_delete_domain(self):
|
||||||
|
context = self.get_context()
|
||||||
|
domain = self.get_domain_fixture()
|
||||||
|
self.backend.delete_domain(context, domain)
|
||||||
|
command = 'NSDCT1 delzone %s\n' % domain['name']
|
||||||
|
self.assertEqual(command, self.server_fixture.server.recved_command)
|
||||||
|
|
||||||
|
def test_server_not_ok(self):
|
||||||
|
self.server_fixture.server.response = 'goat'
|
||||||
|
context = self.get_context()
|
||||||
|
domain = self.get_domain_fixture()
|
||||||
|
self.assertRaises(exceptions.Backend,
|
||||||
|
self.backend.create_domain,
|
||||||
|
context, domain)
|
||||||
|
|
||||||
|
def test_ssl_error(self):
|
||||||
|
self.backend._command = MagicMock(side_effect=ssl.SSLError)
|
||||||
|
context = self.get_context()
|
||||||
|
domain = self.get_domain_fixture()
|
||||||
|
self.assertRaises(exceptions.Backend,
|
||||||
|
self.backend.create_domain,
|
||||||
|
context, domain)
|
||||||
|
|
||||||
|
def test_socket_error(self):
|
||||||
|
self.backend._command = MagicMock(side_effect=socket.error)
|
||||||
|
context = self.get_context()
|
||||||
|
domain = self.get_domain_fixture()
|
||||||
|
self.assertRaises(exceptions.Backend,
|
||||||
|
self.backend.create_domain,
|
||||||
|
context, domain)
|
@ -83,6 +83,7 @@ designate.backend =
|
|||||||
powerdns = designate.backend.impl_powerdns:PowerDNSBackend
|
powerdns = designate.backend.impl_powerdns:PowerDNSBackend
|
||||||
dynect = designate.backend.impl_dynect:DynECTBackend
|
dynect = designate.backend.impl_dynect:DynECTBackend
|
||||||
akamai = designate.backend.impl_akamai:AkamaiBackend
|
akamai = designate.backend.impl_akamai:AkamaiBackend
|
||||||
|
nsd4 = designate.backend.impl_nsd4:NSD4Backend
|
||||||
fake = designate.backend.impl_fake:FakeBackend
|
fake = designate.backend.impl_fake:FakeBackend
|
||||||
agent = designate.backend.agent:AgentPoolBackend
|
agent = designate.backend.agent:AgentPoolBackend
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user