Enable use of Pools YAML
This change adds the tooling to use the DB Tables created for pool config data and the tooling to migrate the config info itself. Change-Id: If99dbf527ef1ac0f05f15fe77f68f64e357fe0a5
This commit is contained in:
parent
1375a45a71
commit
e612a3974f
designate
api/v2/controllers
manage
objects
adapters
__init__.py
pool_also_notify.pypool_nameserver.pypool_target.pypool_target_master.pypool_target_option.pyyaml
pool_manager
tests
devstack
doc/source
etc/designate
releasenotes/notes
@ -16,6 +16,7 @@ import pecan
|
||||
from oslo_log import log as logging
|
||||
|
||||
from designate import utils
|
||||
from designate.i18n import _LW
|
||||
from designate.api.v2.controllers import rest
|
||||
from designate.objects import Pool
|
||||
from designate.objects.adapters import DesignateAdapter
|
||||
@ -63,6 +64,11 @@ class PoolsController(rest.RestController):
|
||||
@pecan.expose(template='json:', content_type='application/json')
|
||||
def post_all(self):
|
||||
"""Create a Pool"""
|
||||
|
||||
LOG.warning(_LW("Use of this API Method is DEPRICATED. This will have "
|
||||
"unforseen side affects when used with the "
|
||||
"designate-manage pool commands"))
|
||||
|
||||
request = pecan.request
|
||||
response = pecan.response
|
||||
context = request.environ['context']
|
||||
@ -86,6 +92,11 @@ class PoolsController(rest.RestController):
|
||||
@utils.validate_uuid('pool_id')
|
||||
def patch_one(self, pool_id):
|
||||
"""Update the specific pool"""
|
||||
|
||||
LOG.warning(_LW("Use of this API Method is DEPRICATED. This will have "
|
||||
"unforseen side affects when used with the "
|
||||
"designate-manage pool commands"))
|
||||
|
||||
request = pecan.request
|
||||
context = request.environ['context']
|
||||
body = request.body_dict
|
||||
@ -111,6 +122,11 @@ class PoolsController(rest.RestController):
|
||||
@utils.validate_uuid('pool_id')
|
||||
def delete_one(self, pool_id):
|
||||
"""Delete the specific pool"""
|
||||
|
||||
LOG.warning(_LW("Use of this API Method is DEPRICATED. This will have "
|
||||
"unforseen side affects when used with the "
|
||||
"designate-manage pool commands"))
|
||||
|
||||
request = pecan.request
|
||||
response = pecan.response
|
||||
context = request.environ['context']
|
||||
|
@ -13,24 +13,172 @@
|
||||
# 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 pprint
|
||||
|
||||
import yaml
|
||||
from oslo_config import cfg
|
||||
from oslo_log import log as logging
|
||||
import oslo_messaging as messaging
|
||||
|
||||
from designate.manage import base
|
||||
from designate import exceptions
|
||||
from designate import rpc
|
||||
from designate.i18n import _LI
|
||||
from designate.i18n import _LC
|
||||
from designate import objects
|
||||
from designate.central import rpcapi as central_rpcapi
|
||||
from designate.manage import base
|
||||
from designate.objects.adapters import DesignateAdapter
|
||||
|
||||
LOG = logging.getLogger(__name__)
|
||||
|
||||
|
||||
CONF = cfg.CONF
|
||||
|
||||
|
||||
class PoolCommands(base.Commands):
|
||||
def __init__(self):
|
||||
super(PoolCommands, self).__init__()
|
||||
rpc.init(cfg.CONF)
|
||||
self.central_api = central_rpcapi.CentralAPI()
|
||||
|
||||
@base.args('--file', help='The path to the file the yaml output should be '
|
||||
'writen to',
|
||||
default='/etc/designate/pools.yaml')
|
||||
def generate_file(self, file):
|
||||
try:
|
||||
pools = self.central_api.find_pools(self.context)
|
||||
except messaging.exceptions.MessagingTimeout:
|
||||
LOG.critical(_LC("No response recieved from designate-central. "
|
||||
"Check it is running, and retry"))
|
||||
with open(file, 'w') as stream:
|
||||
yaml.dump(
|
||||
DesignateAdapter.render('YAML', pools),
|
||||
stream,
|
||||
default_flow_style=False
|
||||
)
|
||||
|
||||
@base.args('--file', help='The path to the file the yaml output should be '
|
||||
'writen to',
|
||||
default='/etc/designate/pools.yaml')
|
||||
def export_from_config(self, file):
|
||||
try:
|
||||
pools = self.central_api.find_pools(self.context)
|
||||
except messaging.exceptions.MessagingTimeout:
|
||||
LOG.critical(_LC("No response recieved from designate-central. "
|
||||
"Check it is running, and retry"))
|
||||
r_pools = objects.PoolList()
|
||||
for pool in pools:
|
||||
r_pool = objects.Pool.from_config(CONF, pool.id)
|
||||
r_pool.id = pool.id
|
||||
r_pool.ns_records = pool.ns_records
|
||||
r_pool.attributes = pool.attributes
|
||||
r_pools.append(r_pool)
|
||||
with open(file, 'w') as stream:
|
||||
yaml.dump(
|
||||
DesignateAdapter.render('YAML', r_pools),
|
||||
stream,
|
||||
default_flow_style=False
|
||||
)
|
||||
|
||||
@base.args('--pool_id', help='ID of the pool to be examined',
|
||||
default=CONF['service:central'].default_pool_id)
|
||||
default=CONF['service:central'].default_pool_id)
|
||||
def show_config(self, pool_id):
|
||||
print('*' * 100)
|
||||
print('Pool Configuration:')
|
||||
print('*' * 100)
|
||||
pprint.pprint(objects.Pool.from_config(CONF, pool_id).to_dict())
|
||||
try:
|
||||
pool = self.central_api.find_pool(self.context, {"id": pool_id})
|
||||
|
||||
print('Pool Configuration:')
|
||||
print('-------------------')
|
||||
|
||||
print(yaml.dump(DesignateAdapter.render('YAML', pool),
|
||||
default_flow_style=False))
|
||||
|
||||
except messaging.exceptions.MessagingTimeout:
|
||||
LOG.critical(_LC("No response recieved from designate-central. "
|
||||
"Check it is running, and retry"))
|
||||
|
||||
@base.args('--file', help='The path to the yaml file describing the pools',
|
||||
default='/etc/designate/pools.yaml')
|
||||
@base.args(
|
||||
'--delete',
|
||||
help='Any Pools not listed in the config file will be deleted. '
|
||||
' WARNING: This will delete any zones left in this pool',
|
||||
default=False)
|
||||
@base.args(
|
||||
'--dry_run',
|
||||
help='This will simulate what will happen when you run this command',
|
||||
default=False)
|
||||
def update(self, file, delete, dry_run):
|
||||
print('Updating Pools Configuration')
|
||||
print('****************************')
|
||||
|
||||
output_msg = ['']
|
||||
|
||||
with open(file, 'r') as stream:
|
||||
xpools = yaml.safe_load(stream)
|
||||
|
||||
if dry_run:
|
||||
output_msg.append("The following changes will occur:")
|
||||
output_msg.append("*********************************")
|
||||
|
||||
for xpool in xpools:
|
||||
try:
|
||||
if 'id' in xpool:
|
||||
try:
|
||||
pool = self.central_api.get_pool(
|
||||
self.context, xpool['id'])
|
||||
except Exception:
|
||||
LOG.critical(
|
||||
_LC("Bad ID Supplied for pool %s"), xpool['name'])
|
||||
continue
|
||||
else:
|
||||
pool = self.central_api.find_pool(
|
||||
self.context, {"name": xpool['name']})
|
||||
|
||||
LOG.info(_LI('Updating existing pool: %s'), pool)
|
||||
|
||||
# TODO(kiall): Move the below into the pool object
|
||||
|
||||
pool = DesignateAdapter.parse('YAML', xpool, pool)
|
||||
|
||||
if dry_run:
|
||||
output_msg.append("Update Pool: %s" % pool)
|
||||
else:
|
||||
pool = self.central_api.update_pool(self.context, pool)
|
||||
|
||||
except exceptions.PoolNotFound:
|
||||
pool = DesignateAdapter.parse('YAML', xpool, objects.Pool())
|
||||
# pool = objects.Pool.from_dict(xpool)
|
||||
if dry_run:
|
||||
output_msg.append("Create Pool: %s" % pool)
|
||||
else:
|
||||
LOG.info(_LI('Creating new pool: %s'), pool)
|
||||
self.central_api.create_pool(self.context, pool)
|
||||
except messaging.exceptions.MessagingTimeout:
|
||||
LOG.critical(_LC("No response recieved from designate-central."
|
||||
" Check it is running, and retry"))
|
||||
|
||||
if delete:
|
||||
pools = self.central_api.find_pools(self.context)
|
||||
pools_in_db = {pool.name for pool in pools}
|
||||
pools_in_yaml = {xpool['name'] for xpool in xpools}
|
||||
|
||||
pools_to_delete = pools_in_db - pools_in_yaml
|
||||
|
||||
for pool in pools_to_delete:
|
||||
try:
|
||||
p = self.central_api.find_pool(
|
||||
self.context,
|
||||
criterion={'name': pool})
|
||||
|
||||
if dry_run:
|
||||
output_msg.append("Delete Pool: %s" % p)
|
||||
|
||||
else:
|
||||
LOG.info(_LI('Deleting %s'), p)
|
||||
self.central_api.delete_pool(self.context, p.id)
|
||||
|
||||
except messaging.exceptions.MessagingTimeout:
|
||||
LOG.critical(_LC("No response recieved from "
|
||||
"designate-central. "
|
||||
"Check it is running, and retry"))
|
||||
|
||||
for line in output_msg:
|
||||
print(line)
|
||||
|
@ -21,7 +21,9 @@ from oslo_db.sqlalchemy.migration_cli import manager as migration_manager
|
||||
from oslo_log import log as logging
|
||||
|
||||
from designate.manage import base
|
||||
from designate import rpc
|
||||
from designate import utils
|
||||
from designate.central import rpcapi as central_rpcapi
|
||||
|
||||
LOG = logging.getLogger(__name__)
|
||||
|
||||
@ -32,33 +34,50 @@ CONF = cfg.CONF
|
||||
utils.register_plugin_opts()
|
||||
|
||||
|
||||
def get_manager(pool_target_id):
|
||||
pool_target_options = CONF['pool_target:%s' % pool_target_id].options
|
||||
connection = pool_target_options['connection']
|
||||
def get_manager(pool_target):
|
||||
connection = pool_target.options.get('connection', None)
|
||||
|
||||
migration_config = {
|
||||
'migration_repo_path': REPOSITORY,
|
||||
'db_url': connection}
|
||||
|
||||
return migration_manager.MigrationManager(migration_config)
|
||||
|
||||
|
||||
class DatabaseCommands(base.Commands):
|
||||
@base.args('pool-target-id', help="Pool Target to Migrate", type=str)
|
||||
def version(self, pool_target_id):
|
||||
current = get_manager(pool_target_id).version()
|
||||
latest = versioning_api.version(repository=REPOSITORY).value
|
||||
print("Current: %s Latest: %s" % (current, latest))
|
||||
def __init__(self):
|
||||
super(DatabaseCommands, self).__init__()
|
||||
rpc.init(cfg.CONF)
|
||||
self.central_api = central_rpcapi.CentralAPI()
|
||||
|
||||
@base.args('pool-target-id', help="Pool Target to Migrate", type=str)
|
||||
def sync(self, pool_target_id):
|
||||
get_manager(pool_target_id).upgrade(None)
|
||||
@base.args('pool-id', help="Pool to Migrate", type=str)
|
||||
def version(self, pool_id):
|
||||
pool = self.central_api.find_pool(self.context, {"id": pool_id})
|
||||
|
||||
@base.args('pool-target-id', help="Pool Target to Migrate", type=str)
|
||||
for pool_target in pool.targets:
|
||||
current = get_manager(pool_target).version()
|
||||
latest = versioning_api.version(repository=REPOSITORY).value
|
||||
print("Current: %s Latest: %s" % (current, latest))
|
||||
|
||||
@base.args('pool-id', help="Pool to Migrate", type=str)
|
||||
def sync(self, pool_id):
|
||||
pool = self.central_api.find_pool(self.context, {"id": pool_id})
|
||||
|
||||
for pool_target in pool.targets:
|
||||
get_manager(pool_target).upgrade(None)
|
||||
|
||||
@base.args('pool-id', help="Pool to Migrate", type=str)
|
||||
@base.args('revision', nargs='?')
|
||||
def upgrade(self, pool_target_id, revision):
|
||||
get_manager(pool_target_id).upgrade(revision)
|
||||
def upgrade(self, pool_id, revision):
|
||||
pool = self.central_api.find_pool(self.context, {"id": pool_id})
|
||||
|
||||
@base.args('pool-target-id', help="Pool Target to Migrate", type=str)
|
||||
for pool_target in pool.targets:
|
||||
get_manager(pool_target).upgrade(revision)
|
||||
|
||||
@base.args('pool-id', help="Pool to Migrate", type=str)
|
||||
@base.args('revision', nargs='?')
|
||||
def downgrade(self, pool_target_id, revision):
|
||||
get_manager(pool_target_id).downgrade(revision)
|
||||
def downgrade(self, pool_id, revision):
|
||||
pool = self.central_api.find_pool(self.context, {"id": pool_id})
|
||||
|
||||
for pool_target in pool.targets:
|
||||
get_manager(pool_target).downgrade(revision)
|
||||
|
@ -32,3 +32,14 @@ from designate.objects.adapters.api_v2.zone_transfer_request import ZoneTransfer
|
||||
from designate.objects.adapters.api_v2.validation_error import ValidationErrorAPIv2Adapter, ValidationErrorListAPIv2Adapter # noqa
|
||||
from designate.objects.adapters.api_v2.zone_import import ZoneImportAPIv2Adapter, ZoneImportListAPIv2Adapter # noqa
|
||||
from designate.objects.adapters.api_v2.zone_export import ZoneExportAPIv2Adapter, ZoneExportListAPIv2Adapter # noqa
|
||||
|
||||
# YAML
|
||||
|
||||
from designate.objects.adapters.yaml.pool import PoolYAMLAdapter, PoolListYAMLAdapter # noqa
|
||||
from designate.objects.adapters.yaml.pool_attribute import PoolAttributeYAMLAdapter, PoolAttributeListYAMLAdapter # noqa
|
||||
from designate.objects.adapters.yaml.pool_also_notify import PoolAlsoNotifyYAMLAdapter, PoolAlsoNotifyListYAMLAdapter # noqa
|
||||
from designate.objects.adapters.yaml.pool_nameserver import PoolNameserverYAMLAdapter, PoolNameserverListYAMLAdapter # noqa
|
||||
from designate.objects.adapters.yaml.pool_ns_record import PoolNsRecordYAMLAdapter, PoolNsRecordListYAMLAdapter # noqa
|
||||
from designate.objects.adapters.yaml.pool_target import PoolTargetYAMLAdapter, PoolTargetListYAMLAdapter # noqa
|
||||
from designate.objects.adapters.yaml.pool_target_master import PoolTargetMasterYAMLAdapter, PoolTargetMasterListYAMLAdapter # noqa
|
||||
from designate.objects.adapters.yaml.pool_target_option import PoolTargetOptionYAMLAdapter, PoolTargetOptionListYAMLAdapter # noqa
|
||||
|
0
designate/objects/adapters/yaml/__init__.py
Normal file
0
designate/objects/adapters/yaml/__init__.py
Normal file
81
designate/objects/adapters/yaml/base.py
Normal file
81
designate/objects/adapters/yaml/base.py
Normal file
@ -0,0 +1,81 @@
|
||||
# Copyright 2016 Hewlett Packard Enterprise Development Company, L.P.
|
||||
#
|
||||
# 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 oslo_log import log as logging
|
||||
|
||||
from designate.objects.adapters import base
|
||||
|
||||
LOG = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class YAMLAdapter(base.DesignateAdapter):
|
||||
|
||||
ADAPTER_FORMAT = 'YAML'
|
||||
|
||||
#####################
|
||||
# Parsing methods #
|
||||
#####################
|
||||
|
||||
@classmethod
|
||||
def parse(cls, values, output_object, *args, **kwargs):
|
||||
obj = super(YAMLAdapter, cls).parse(
|
||||
cls.ADAPTER_FORMAT, values, output_object, *args, **kwargs)
|
||||
return obj
|
||||
|
||||
@classmethod
|
||||
def _render_object(cls, object, *args, **kwargs):
|
||||
# The dict we will return to be rendered to JSON / output format
|
||||
r_obj = {}
|
||||
# Loop over all fields that are supposed to be output
|
||||
for key, value in cls.MODIFICATIONS['fields'].items():
|
||||
# Get properties for this field
|
||||
field_props = cls.MODIFICATIONS['fields'][key]
|
||||
# Check if it has to be renamed
|
||||
if field_props.get('rename', False):
|
||||
obj = getattr(object, field_props.get('rename'))
|
||||
# if rename is specified we need to change the key
|
||||
obj_key = field_props.get('rename')
|
||||
else:
|
||||
# if not, move on
|
||||
obj = getattr(object, key, None)
|
||||
obj_key = key
|
||||
# Check if this item is a relation (another DesignateObject that
|
||||
# will need to be converted itself
|
||||
if object.FIELDS.get(obj_key, {}).get('relation'):
|
||||
# Get a adapter for the nested object
|
||||
# Get the class the object is and get its adapter, then set
|
||||
# the item in the dict to the output
|
||||
r_obj[key] = cls.get_object_adapter(
|
||||
cls.ADAPTER_FORMAT,
|
||||
object.FIELDS[obj_key].get('relation_cls')).render(
|
||||
cls.ADAPTER_FORMAT, obj, *args, **kwargs)
|
||||
elif object.FIELDS.get(
|
||||
obj_key, {}).get('schema', {}).get('type') == 'integer':
|
||||
r_obj[key] = int(obj)
|
||||
elif obj is not None:
|
||||
# Just attach the damn item if there is no weird edge cases
|
||||
r_obj[key] = str(obj)
|
||||
# Send it back
|
||||
return r_obj
|
||||
|
||||
@classmethod
|
||||
def _render_list(cls, list_object, *args, **kwargs):
|
||||
# The list we will return to be rendered to JSON / output format
|
||||
r_list = []
|
||||
# iterate and convert each DesignateObject in the list, and append to
|
||||
# the object we are returning
|
||||
for object in list_object:
|
||||
r_list.append(cls.get_object_adapter(
|
||||
cls.ADAPTER_FORMAT,
|
||||
object).render(cls.ADAPTER_FORMAT, object, *args, **kwargs))
|
||||
return r_list
|
59
designate/objects/adapters/yaml/pool.py
Normal file
59
designate/objects/adapters/yaml/pool.py
Normal file
@ -0,0 +1,59 @@
|
||||
# Copyright 2014 Hewlett-Packard Development Company, L.P.
|
||||
#
|
||||
# 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 oslo_log import log as logging
|
||||
|
||||
from designate.objects.adapters.yaml import base
|
||||
from designate import objects
|
||||
LOG = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class PoolYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.Pool
|
||||
|
||||
MODIFICATIONS = {
|
||||
'fields': {
|
||||
'id': {
|
||||
'read_only': False
|
||||
},
|
||||
'name': {
|
||||
'read_only': False
|
||||
},
|
||||
'description': {
|
||||
'read_only': False
|
||||
},
|
||||
'attributes': {
|
||||
'read_only': False
|
||||
},
|
||||
'ns_records': {
|
||||
'read_only': False
|
||||
},
|
||||
'nameservers': {
|
||||
'read_only': False
|
||||
},
|
||||
'targets': {
|
||||
'read_only': False
|
||||
},
|
||||
'also_notifies': {
|
||||
'read_only': False
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
class PoolListYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.PoolList
|
||||
|
||||
MODIFICATIONS = {}
|
41
designate/objects/adapters/yaml/pool_also_notify.py
Normal file
41
designate/objects/adapters/yaml/pool_also_notify.py
Normal file
@ -0,0 +1,41 @@
|
||||
# Copyright 2014 Hewlett-Packard Development Company, L.P.
|
||||
#
|
||||
# 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 oslo_log import log as logging
|
||||
|
||||
from designate.objects.adapters.yaml import base
|
||||
from designate import objects
|
||||
LOG = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class PoolAlsoNotifyYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.PoolAlsoNotify
|
||||
|
||||
MODIFICATIONS = {
|
||||
'fields': {
|
||||
'host': {
|
||||
'read_only': False
|
||||
},
|
||||
'port': {
|
||||
'read_only': False
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
class PoolAlsoNotifyListYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.PoolAlsoNotifyList
|
||||
|
||||
MODIFICATIONS = {}
|
86
designate/objects/adapters/yaml/pool_attribute.py
Normal file
86
designate/objects/adapters/yaml/pool_attribute.py
Normal file
@ -0,0 +1,86 @@
|
||||
# Copyright 2014 Hewlett-Packard Development Company, L.P.
|
||||
#
|
||||
# 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 oslo_log import log as logging
|
||||
|
||||
from designate.objects.adapters.yaml import base
|
||||
from designate import objects
|
||||
LOG = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class PoolAttributeYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.PoolAttribute
|
||||
|
||||
MODIFICATIONS = {
|
||||
'fields': {
|
||||
'key': {
|
||||
'read_only': False
|
||||
},
|
||||
'value': {
|
||||
'read_only': False
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
@classmethod
|
||||
def _render_object(cls, object, *arg, **kwargs):
|
||||
return {str(object.key): str(object.value)}
|
||||
|
||||
@classmethod
|
||||
def _parse_object(cls, values, object, *args, **kwargs):
|
||||
for key in six.iterkeys(values):
|
||||
object.key = key
|
||||
object.value = values[key]
|
||||
|
||||
return object
|
||||
|
||||
|
||||
class PoolAttributeListYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.PoolAttributeList
|
||||
|
||||
MODIFICATIONS = {}
|
||||
|
||||
@classmethod
|
||||
def _render_list(cls, list_object, *args, **kwargs):
|
||||
|
||||
r_list = {}
|
||||
|
||||
for object in list_object:
|
||||
value = cls.get_object_adapter(
|
||||
cls.ADAPTER_FORMAT,
|
||||
object).render(cls.ADAPTER_FORMAT, object, *args, **kwargs)
|
||||
for key in six.iterkeys(value):
|
||||
r_list[key] = value[key]
|
||||
|
||||
return r_list
|
||||
|
||||
@classmethod
|
||||
def _parse_list(cls, values, output_object, *args, **kwargs):
|
||||
|
||||
for key, value in values.items():
|
||||
# Add the object to the list
|
||||
output_object.append(
|
||||
# Get the right Adapter
|
||||
cls.get_object_adapter(
|
||||
cls.ADAPTER_FORMAT,
|
||||
# This gets the internal type of the list, and parses it
|
||||
# We need to do `get_object_adapter` as we need a new
|
||||
# instance of the Adapter
|
||||
output_object.LIST_ITEM_TYPE()).parse(
|
||||
{key: value}, output_object.LIST_ITEM_TYPE()))
|
||||
|
||||
# Return the filled list
|
||||
return output_object
|
41
designate/objects/adapters/yaml/pool_nameserver.py
Normal file
41
designate/objects/adapters/yaml/pool_nameserver.py
Normal file
@ -0,0 +1,41 @@
|
||||
# Copyright 2014 Hewlett-Packard Development Company, L.P.
|
||||
#
|
||||
# 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 oslo_log import log as logging
|
||||
|
||||
from designate.objects.adapters.yaml import base
|
||||
from designate import objects
|
||||
LOG = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class PoolNameserverYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.PoolNameserver
|
||||
|
||||
MODIFICATIONS = {
|
||||
'fields': {
|
||||
'host': {
|
||||
'read_only': False
|
||||
},
|
||||
'port': {
|
||||
'read_only': False
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
class PoolNameserverListYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.PoolNameserverList
|
||||
|
||||
MODIFICATIONS = {}
|
41
designate/objects/adapters/yaml/pool_ns_record.py
Normal file
41
designate/objects/adapters/yaml/pool_ns_record.py
Normal file
@ -0,0 +1,41 @@
|
||||
# Copyright 2014 Hewlett-Packard Development Company, L.P.
|
||||
#
|
||||
# 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 oslo_log import log as logging
|
||||
|
||||
from designate.objects.adapters.yaml import base
|
||||
from designate import objects
|
||||
LOG = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class PoolNsRecordYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.PoolNsRecord
|
||||
|
||||
MODIFICATIONS = {
|
||||
'fields': {
|
||||
'priority': {
|
||||
'read_only': False
|
||||
},
|
||||
'hostname': {
|
||||
'read_only': False
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
class PoolNsRecordListYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.PoolNsRecordList
|
||||
|
||||
MODIFICATIONS = {}
|
50
designate/objects/adapters/yaml/pool_target.py
Normal file
50
designate/objects/adapters/yaml/pool_target.py
Normal file
@ -0,0 +1,50 @@
|
||||
# Copyright 2014 Hewlett-Packard Development Company, L.P.
|
||||
#
|
||||
# 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 oslo_log import log as logging
|
||||
|
||||
from designate.objects.adapters.yaml import base
|
||||
from designate import objects
|
||||
LOG = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class PoolTargetYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.PoolTarget
|
||||
|
||||
MODIFICATIONS = {
|
||||
'fields': {
|
||||
'type': {
|
||||
'read_only': False
|
||||
},
|
||||
'tsigkey_id': {
|
||||
'read_only': False
|
||||
},
|
||||
'description': {
|
||||
'read_only': False
|
||||
},
|
||||
'masters': {
|
||||
'read_only': False
|
||||
},
|
||||
'options': {
|
||||
'read_only': False
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
class PoolTargetListYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.PoolTargetList
|
||||
|
||||
MODIFICATIONS = {}
|
41
designate/objects/adapters/yaml/pool_target_master.py
Normal file
41
designate/objects/adapters/yaml/pool_target_master.py
Normal file
@ -0,0 +1,41 @@
|
||||
# Copyright 2014 Hewlett-Packard Development Company, L.P.
|
||||
#
|
||||
# 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 oslo_log import log as logging
|
||||
|
||||
from designate.objects.adapters.yaml import base
|
||||
from designate import objects
|
||||
LOG = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class PoolTargetMasterYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.PoolTargetMaster
|
||||
|
||||
MODIFICATIONS = {
|
||||
'fields': {
|
||||
'host': {
|
||||
'read_only': False
|
||||
},
|
||||
'port': {
|
||||
'read_only': False
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
class PoolTargetMasterListYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.PoolTargetMasterList
|
||||
|
||||
MODIFICATIONS = {}
|
86
designate/objects/adapters/yaml/pool_target_option.py
Normal file
86
designate/objects/adapters/yaml/pool_target_option.py
Normal file
@ -0,0 +1,86 @@
|
||||
# Copyright 2014 Hewlett-Packard Development Company, L.P.
|
||||
#
|
||||
# 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 oslo_log import log as logging
|
||||
|
||||
from designate.objects.adapters.yaml import base
|
||||
from designate import objects
|
||||
LOG = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class PoolTargetOptionYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.PoolTargetOption
|
||||
|
||||
MODIFICATIONS = {
|
||||
'fields': {
|
||||
'key': {
|
||||
'read_only': False
|
||||
},
|
||||
'value': {
|
||||
'read_only': False
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
@classmethod
|
||||
def _render_object(cls, object, *arg, **kwargs):
|
||||
return {str(object.key): str(object.value)}
|
||||
|
||||
@classmethod
|
||||
def _parse_object(cls, values, object, *args, **kwargs):
|
||||
for key in six.iterkeys(values):
|
||||
object.key = key
|
||||
object.value = values[key]
|
||||
|
||||
return object
|
||||
|
||||
|
||||
class PoolTargetOptionListYAMLAdapter(base.YAMLAdapter):
|
||||
|
||||
ADAPTER_OBJECT = objects.PoolTargetOptionList
|
||||
|
||||
MODIFICATIONS = {}
|
||||
|
||||
@classmethod
|
||||
def _render_list(cls, list_object, *args, **kwargs):
|
||||
|
||||
r_list = {}
|
||||
|
||||
for object in list_object:
|
||||
value = cls.get_object_adapter(
|
||||
cls.ADAPTER_FORMAT,
|
||||
object).render(cls.ADAPTER_FORMAT, object, *args, **kwargs)
|
||||
for key in six.iterkeys(value):
|
||||
r_list[key] = value[key]
|
||||
|
||||
return r_list
|
||||
|
||||
@classmethod
|
||||
def _parse_list(cls, values, output_object, *args, **kwargs):
|
||||
|
||||
for key, value in values.items():
|
||||
# Add the object to the list
|
||||
output_object.append(
|
||||
# Get the right Adapter
|
||||
cls.get_object_adapter(
|
||||
cls.ADAPTER_FORMAT,
|
||||
# This gets the internal type of the list, and parses it
|
||||
# We need to do `get_object_adapter` as we need a new
|
||||
# instance of the Adapter
|
||||
output_object.LIST_ITEM_TYPE()).parse(
|
||||
{key: value}, output_object.LIST_ITEM_TYPE()))
|
||||
|
||||
# Return the filled list
|
||||
return output_object
|
@ -18,11 +18,34 @@ from designate.objects import base
|
||||
class PoolAlsoNotify(base.DictObjectMixin, base.PersistentObjectMixin,
|
||||
base.DesignateObject):
|
||||
FIELDS = {
|
||||
'pool_id': {},
|
||||
'host': {},
|
||||
'port': {},
|
||||
'pool_id': {
|
||||
'schema': {
|
||||
'type': 'string',
|
||||
'description': 'Pool identifier',
|
||||
'format': 'uuid',
|
||||
},
|
||||
},
|
||||
'host': {
|
||||
'schema': {
|
||||
'type': 'string',
|
||||
'format': 'ip-or-host',
|
||||
'required': True,
|
||||
},
|
||||
},
|
||||
'port': {
|
||||
'schema': {
|
||||
'type': 'integer',
|
||||
'minimum': 1,
|
||||
'maximum': 65535,
|
||||
'required': True,
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
STRING_KEYS = [
|
||||
'id', 'host', 'port', 'pool_id'
|
||||
]
|
||||
|
||||
|
||||
class PoolAlsoNotifyList(base.ListObjectMixin, base.DesignateObject):
|
||||
LIST_ITEM_TYPE = PoolAlsoNotify
|
||||
|
@ -18,11 +18,34 @@ from designate.objects import base
|
||||
class PoolNameserver(base.DictObjectMixin, base.PersistentObjectMixin,
|
||||
base.DesignateObject):
|
||||
FIELDS = {
|
||||
'pool_id': {},
|
||||
'host': {},
|
||||
'port': {},
|
||||
'pool_id': {
|
||||
'schema': {
|
||||
'type': 'string',
|
||||
'description': 'Pool identifier',
|
||||
'format': 'uuid',
|
||||
},
|
||||
},
|
||||
'host': {
|
||||
'schema': {
|
||||
'type': 'string',
|
||||
'format': 'ip-or-host',
|
||||
'required': True,
|
||||
},
|
||||
},
|
||||
'port': {
|
||||
'schema': {
|
||||
'type': 'integer',
|
||||
'minimum': 1,
|
||||
'maximum': 65535,
|
||||
'required': True,
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
STRING_KEYS = [
|
||||
'id', 'host', 'port', 'pool_id'
|
||||
]
|
||||
|
||||
|
||||
class PoolNameserverList(base.ListObjectMixin, base.DesignateObject):
|
||||
LIST_ITEM_TYPE = PoolNameserver
|
||||
|
@ -18,11 +18,28 @@ from designate.objects import base
|
||||
class PoolTarget(base.DictObjectMixin, base.PersistentObjectMixin,
|
||||
base.DesignateObject):
|
||||
FIELDS = {
|
||||
'pool_id': {},
|
||||
'pool_id': {
|
||||
'schema': {
|
||||
'type': 'string',
|
||||
'description': 'Pool identifier',
|
||||
'format': 'uuid',
|
||||
},
|
||||
},
|
||||
'type': {},
|
||||
'tsigkey_id': {},
|
||||
'description': {},
|
||||
|
||||
'tsigkey_id': {
|
||||
'schema': {
|
||||
'type': ['string', 'null'],
|
||||
'description': 'TSIG identifier',
|
||||
'format': 'uuid',
|
||||
},
|
||||
},
|
||||
'description': {
|
||||
'schema': {
|
||||
'type': ['string', 'null'],
|
||||
'description': 'Description for the pool',
|
||||
'maxLength': 160
|
||||
}
|
||||
},
|
||||
'masters': {
|
||||
'relation': True,
|
||||
'relation_cls': 'PoolTargetMasterList'
|
||||
@ -33,6 +50,10 @@ class PoolTarget(base.DictObjectMixin, base.PersistentObjectMixin,
|
||||
},
|
||||
}
|
||||
|
||||
STRING_KEYS = [
|
||||
'id', 'type', 'pool_id'
|
||||
]
|
||||
|
||||
|
||||
class PoolTargetList(base.ListObjectMixin, base.DesignateObject):
|
||||
LIST_ITEM_TYPE = PoolTarget
|
||||
|
@ -18,11 +18,34 @@ from designate.objects import base
|
||||
class PoolTargetMaster(base.DictObjectMixin, base.PersistentObjectMixin,
|
||||
base.DesignateObject):
|
||||
FIELDS = {
|
||||
'pool_target_id': {},
|
||||
'host': {},
|
||||
'port': {}
|
||||
'pool_target_id': {
|
||||
'schema': {
|
||||
'type': 'string',
|
||||
'description': 'Pool Target identifier',
|
||||
'format': 'uuid',
|
||||
},
|
||||
},
|
||||
'host': {
|
||||
'schema': {
|
||||
'type': 'string',
|
||||
'format': 'ip-or-host',
|
||||
'required': True,
|
||||
},
|
||||
},
|
||||
'port': {
|
||||
'schema': {
|
||||
'type': 'integer',
|
||||
'minimum': 1,
|
||||
'maximum': 65535,
|
||||
'required': True,
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
STRING_KEYS = [
|
||||
'id', 'host', 'port', 'pool_target_id'
|
||||
]
|
||||
|
||||
|
||||
class PoolTargetMasterList(base.ListObjectMixin, base.DesignateObject):
|
||||
LIST_ITEM_TYPE = PoolTargetMaster
|
||||
|
@ -18,11 +18,33 @@ from designate.objects import base
|
||||
class PoolTargetOption(base.DictObjectMixin, base.PersistentObjectMixin,
|
||||
base.DesignateObject):
|
||||
FIELDS = {
|
||||
'pool_target_id': {},
|
||||
'key': {},
|
||||
'value': {},
|
||||
'pool_target_id': {
|
||||
'schema': {
|
||||
'type': 'string',
|
||||
'description': 'Pool Target identifier',
|
||||
'format': 'uuid',
|
||||
},
|
||||
},
|
||||
'key': {
|
||||
'schema': {
|
||||
'type': 'string',
|
||||
'maxLength': 255,
|
||||
},
|
||||
'required': True,
|
||||
},
|
||||
'value': {
|
||||
'schema': {
|
||||
'type': 'string',
|
||||
'maxLength': 255,
|
||||
},
|
||||
'required': True
|
||||
}
|
||||
}
|
||||
|
||||
STRING_KEYS = [
|
||||
'id', 'key', 'value', 'pool_target_id'
|
||||
]
|
||||
|
||||
|
||||
class PoolTargetOptionList(base.AttributeListObjectMixin,
|
||||
base.DesignateObject):
|
||||
|
@ -97,10 +97,6 @@ class Service(service.RPCService, coordination.CoordinationMixin,
|
||||
def __init__(self, threads=None):
|
||||
super(Service, self).__init__(threads=threads)
|
||||
|
||||
# Build the Pool (and related) Object from Config
|
||||
self.pool = objects.Pool.from_config(
|
||||
CONF, CONF['service:pool_manager'].pool_id)
|
||||
|
||||
# Get a pool manager cache connection.
|
||||
self.cache = cache.get_pool_manager_cache(
|
||||
CONF['service:pool_manager'].cache_driver)
|
||||
@ -118,11 +114,8 @@ class Service(service.RPCService, coordination.CoordinationMixin,
|
||||
|
||||
# Compute a time (seconds) by which things should have propagated
|
||||
self.max_prop_time = (self.timeout * self.max_retries +
|
||||
self.max_retries * self.retry_interval +
|
||||
self.delay)
|
||||
|
||||
# Create the necessary Backend instances for each target
|
||||
self._setup_target_backends()
|
||||
self.max_retries * self.retry_interval +
|
||||
self.delay)
|
||||
|
||||
def _setup_target_backends(self):
|
||||
self.target_backends = {}
|
||||
@ -154,6 +147,34 @@ class Service(service.RPCService, coordination.CoordinationMixin,
|
||||
return topic
|
||||
|
||||
def start(self):
|
||||
|
||||
# Build the Pool (and related) Object from Config
|
||||
context = DesignateContext.get_admin_context()
|
||||
pool_id = CONF['service:pool_manager'].pool_id
|
||||
|
||||
has_targets = False
|
||||
|
||||
while not has_targets:
|
||||
try:
|
||||
self.pool = self.central_api.get_pool(context, pool_id)
|
||||
|
||||
if len(self.pool.targets) > 0:
|
||||
has_targets = True
|
||||
else:
|
||||
LOG.error(_LE("No targets for %s found."), self.pool)
|
||||
time.sleep(5)
|
||||
|
||||
# Pool data may not have migrated to the DB yet
|
||||
except exceptions.PoolNotFound:
|
||||
LOG.error(_LE("Pool ID %s not found."), pool_id)
|
||||
time.sleep(5)
|
||||
# designate-central service may not have started yet
|
||||
except messaging.exceptions.MessagingTimeout:
|
||||
time.sleep(0.2)
|
||||
|
||||
# Create the necessary Backend instances for each target
|
||||
self._setup_target_backends()
|
||||
|
||||
for target in self.pool.targets:
|
||||
self.target_backends[target.id].start()
|
||||
|
||||
|
@ -382,9 +382,14 @@ class TestCase(base.BaseTestCase):
|
||||
# Fetch the default pool
|
||||
pool = self.storage.get_pool(self.admin_context, default_pool_id)
|
||||
|
||||
# Add a NS record to it
|
||||
pool.ns_records.append(
|
||||
objects.PoolNsRecord(priority=0, hostname='ns1.example.org.'))
|
||||
# Fill out the necessary pool details
|
||||
pool.ns_records = objects.PoolNsRecordList.from_list([
|
||||
{'hostname': 'ns1.example.org.', 'priority': 0}
|
||||
])
|
||||
|
||||
pool.targets = objects.PoolTargetList.from_list([
|
||||
{'type': 'fake', u'description': "Fake PoolTarget for Tests"}
|
||||
])
|
||||
|
||||
# Save the default pool
|
||||
self.storage.update_pool(self.admin_context, pool)
|
||||
|
63
designate/tests/resources/pools_yaml/multiple-pools.yaml
Normal file
63
designate/tests/resources/pools_yaml/multiple-pools.yaml
Normal file
@ -0,0 +1,63 @@
|
||||
---
|
||||
- name: pool-1
|
||||
description: Default PowerDNS Pool
|
||||
attributes:
|
||||
internal: true
|
||||
ns_records:
|
||||
- hostname: ns1-1.example.org.
|
||||
priority: 1
|
||||
- hostname: ns1-2.example.org.
|
||||
priority: 2
|
||||
nameservers:
|
||||
- host: 192.0.2.2
|
||||
port: 53
|
||||
- host: 192.0.2.3
|
||||
port: 53
|
||||
targets:
|
||||
- type: powerdns
|
||||
description: PowerDNS Database Cluster
|
||||
masters:
|
||||
- host: 192.0.2.1
|
||||
port: 5354
|
||||
options:
|
||||
connection: 'mysql+pymysql://designate:password@127.0.0.1/designate_pdns?charset=utf8'
|
||||
also_notifies:
|
||||
- host: 192.0.2.4
|
||||
port: 53
|
||||
|
||||
- name: pool-2
|
||||
id: cf2e8eab-76cd-4162-bf76-8aeee3556de0
|
||||
description: Default PowerDNS Pool
|
||||
|
||||
attributes:
|
||||
external: true
|
||||
ns_records:
|
||||
- hostname: ns1-1.example.org.
|
||||
priority: 1
|
||||
- hostname: ns1-2.example.org.
|
||||
priority: 2
|
||||
nameservers:
|
||||
- host: 192.0.2.2
|
||||
port: 53
|
||||
- host: 192.0.2.3
|
||||
port: 53
|
||||
|
||||
targets:
|
||||
- type: bind
|
||||
description: BIND9 Server 1
|
||||
masters:
|
||||
- host: 192.0.2.1
|
||||
port: 5354
|
||||
options:
|
||||
rndc_host: 192.0.2.2
|
||||
rndc_port: 953
|
||||
rndc_key_file: /etc/designate/rndc.key
|
||||
- type: bind
|
||||
description: BIND9 Server 2
|
||||
masters:
|
||||
- host: 192.0.2.1
|
||||
port: 5354
|
||||
options:
|
||||
rndc_host: 192.0.2.3
|
||||
rndc_port: 953
|
||||
rndc_key_file: /etc/designate/rndc.key
|
32
designate/tests/resources/pools_yaml/pools.yaml
Normal file
32
designate/tests/resources/pools_yaml/pools.yaml
Normal file
@ -0,0 +1,32 @@
|
||||
---
|
||||
|
||||
- name: default
|
||||
description: Default PowerDNS Pool
|
||||
|
||||
attributes:
|
||||
type: internal
|
||||
|
||||
ns_records:
|
||||
- hostname: ns1-1.example.org.
|
||||
priority: 1
|
||||
- hostname: ns1-2.example.org.
|
||||
priority: 2
|
||||
|
||||
nameservers:
|
||||
- host: 192.0.2.2
|
||||
port: 53
|
||||
- host: 192.0.2.3
|
||||
port: 53
|
||||
|
||||
targets:
|
||||
- type: powerdns
|
||||
description: PowerDNS Database Cluster
|
||||
masters:
|
||||
- host: 192.0.2.1
|
||||
port: 5354
|
||||
options:
|
||||
connection: 'mysql+pymysql://designate:password@127.0.0.1/designate_pdns?charset=utf8'
|
||||
also_notifies:
|
||||
- host: 192.0.2.4
|
||||
port: 53
|
||||
|
25
designate/tests/resources/pools_yaml/sample_output.yaml
Normal file
25
designate/tests/resources/pools_yaml/sample_output.yaml
Normal file
@ -0,0 +1,25 @@
|
||||
- also_notifies:
|
||||
- host: 192.0.2.4
|
||||
port: 53
|
||||
attributes: {}
|
||||
description: Default PowerDNS Pool
|
||||
id: cf2e8eab-76cd-4162-bf76-8aeee3556de0
|
||||
name: default
|
||||
nameservers:
|
||||
- host: 192.0.2.2
|
||||
port: 53
|
||||
- host: 192.0.2.3
|
||||
port: 53
|
||||
ns_records:
|
||||
- hostname: ns1-1.example.org.
|
||||
priority: 1
|
||||
- hostname: ns1-2.example.org.
|
||||
priority: 2
|
||||
targets:
|
||||
- description: PowerDNS Database Cluster
|
||||
masters:
|
||||
- host: 192.0.2.1
|
||||
port: 5354
|
||||
options:
|
||||
connection: mysql+pymysql://designate:password@127.0.0.1/designate_pdns?charset=utf8
|
||||
type: powerdns
|
@ -15,6 +15,49 @@
|
||||
# under the License.
|
||||
from designate.tests import TestCase
|
||||
|
||||
POOL_DICT = {
|
||||
'id': u'794ccc2c-d751-44fe-b57f-8894c9f5c842',
|
||||
'name': u'default',
|
||||
'targets': [
|
||||
{
|
||||
'id': 'f278782a-07dc-4502-9177-b5d85c5f7c7e',
|
||||
'type': 'fake',
|
||||
'masters': [
|
||||
{
|
||||
'host': '127.0.0.1',
|
||||
'port': 5354
|
||||
}
|
||||
],
|
||||
'options': {}
|
||||
},
|
||||
{
|
||||
'id': 'a38703f2-b71e-4e5b-ab22-30caaed61dfd',
|
||||
'type': 'fake',
|
||||
'masters': [
|
||||
{
|
||||
'host': '127.0.0.1',
|
||||
'port': 5354
|
||||
}
|
||||
],
|
||||
'options': {}
|
||||
},
|
||||
],
|
||||
'nameservers': [
|
||||
{
|
||||
'id': 'c5d64303-4cba-425a-9f3c-5d708584dde4',
|
||||
'host': '127.0.0.1',
|
||||
'port': 5355
|
||||
|
||||
},
|
||||
{
|
||||
'id': 'c67cdc95-9a9e-4d2a-98ed-dc78cbd85234',
|
||||
'host': '127.0.0.1',
|
||||
'port': 5356
|
||||
},
|
||||
],
|
||||
'also_notifies': [],
|
||||
}
|
||||
|
||||
|
||||
class PoolManagerTestCase(TestCase):
|
||||
pass
|
||||
|
@ -18,7 +18,6 @@ import logging
|
||||
import uuid
|
||||
|
||||
import oslo_messaging as messaging
|
||||
from oslo_config import cfg
|
||||
from mock import call
|
||||
from mock import Mock
|
||||
from mock import patch
|
||||
@ -30,6 +29,7 @@ from designate.central import rpcapi as central_rpcapi
|
||||
from designate.mdns import rpcapi as mdns_rpcapi
|
||||
from designate.storage.impl_sqlalchemy import tables
|
||||
from designate.tests.test_pool_manager import PoolManagerTestCase
|
||||
from designate.tests.test_pool_manager import POOL_DICT
|
||||
import designate.pool_manager.service as pm_module
|
||||
|
||||
LOG = logging.getLogger(__name__)
|
||||
@ -55,61 +55,13 @@ class PoolManagerServiceNoopTest(PoolManagerTestCase):
|
||||
pool_id='794ccc2c-d751-44fe-b57f-8894c9f5c842',
|
||||
group='service:pool_manager')
|
||||
|
||||
# Configure the Pool
|
||||
section_name = 'pool:794ccc2c-d751-44fe-b57f-8894c9f5c842'
|
||||
section_opts = [
|
||||
cfg.ListOpt('targets', default=[
|
||||
'f278782a-07dc-4502-9177-b5d85c5f7c7e',
|
||||
'a38703f2-b71e-4e5b-ab22-30caaed61dfd',
|
||||
]),
|
||||
cfg.ListOpt('nameservers', default=[
|
||||
'c5d64303-4cba-425a-9f3c-5d708584dde4',
|
||||
'c67cdc95-9a9e-4d2a-98ed-dc78cbd85234',
|
||||
]),
|
||||
cfg.ListOpt('also_notifies', default=[]),
|
||||
]
|
||||
cfg.CONF.register_group(cfg.OptGroup(name=section_name))
|
||||
cfg.CONF.register_opts(section_opts, group=section_name)
|
||||
|
||||
# Configure the Pool Targets
|
||||
section_name = 'pool_target:f278782a-07dc-4502-9177-b5d85c5f7c7e'
|
||||
section_opts = [
|
||||
cfg.StrOpt('type', default='fake'),
|
||||
cfg.ListOpt('masters', default=['127.0.0.1:5354']),
|
||||
cfg.DictOpt('options', default={})
|
||||
]
|
||||
cfg.CONF.register_group(cfg.OptGroup(name=section_name))
|
||||
cfg.CONF.register_opts(section_opts, group=section_name)
|
||||
|
||||
section_name = 'pool_target:a38703f2-b71e-4e5b-ab22-30caaed61dfd'
|
||||
section_opts = [
|
||||
cfg.StrOpt('type', default='fake'),
|
||||
cfg.ListOpt('masters', default=['127.0.0.1:5354']),
|
||||
cfg.DictOpt('options', default={})
|
||||
]
|
||||
cfg.CONF.register_group(cfg.OptGroup(name=section_name))
|
||||
cfg.CONF.register_opts(section_opts, group=section_name)
|
||||
|
||||
# Configure the Pool Nameservers
|
||||
section_name = 'pool_nameserver:c5d64303-4cba-425a-9f3c-5d708584dde4'
|
||||
section_opts = [
|
||||
cfg.StrOpt('host', default='127.0.0.1'),
|
||||
cfg.StrOpt('port', default=5355),
|
||||
]
|
||||
cfg.CONF.register_group(cfg.OptGroup(name=section_name))
|
||||
cfg.CONF.register_opts(section_opts, group=section_name)
|
||||
|
||||
section_name = 'pool_nameserver:c67cdc95-9a9e-4d2a-98ed-dc78cbd85234'
|
||||
section_opts = [
|
||||
cfg.StrOpt('host', default='127.0.0.1'),
|
||||
cfg.StrOpt('port', default=5356),
|
||||
]
|
||||
cfg.CONF.register_group(cfg.OptGroup(name=section_name))
|
||||
cfg.CONF.register_opts(section_opts, group=section_name)
|
||||
|
||||
# Start the Service
|
||||
self.service = self.start_service('pool_manager')
|
||||
self.cache = self.service.cache
|
||||
with patch.object(
|
||||
central_rpcapi.CentralAPI,
|
||||
'get_pool',
|
||||
return_value=objects.Pool.from_dict(POOL_DICT)):
|
||||
self.service = self.start_service('pool_manager')
|
||||
self.cache = self.service.cache
|
||||
|
||||
@staticmethod
|
||||
def _build_zone(name, action, status, id=None):
|
||||
@ -230,7 +182,11 @@ class PoolManagerServiceNoopTest(PoolManagerTestCase):
|
||||
self.config(
|
||||
threshold_percentage=50,
|
||||
group='service:pool_manager')
|
||||
self.service = self.start_service('pool_manager')
|
||||
with patch.object(
|
||||
central_rpcapi.CentralAPI,
|
||||
'get_pool',
|
||||
return_value=objects.Pool.from_dict(POOL_DICT)):
|
||||
self.service = self.start_service('pool_manager')
|
||||
|
||||
zone = self._build_zone('example.org.', 'CREATE', 'PENDING')
|
||||
|
||||
@ -349,7 +305,11 @@ class PoolManagerServiceNoopTest(PoolManagerTestCase):
|
||||
self.config(
|
||||
threshold_percentage=50,
|
||||
group='service:pool_manager')
|
||||
self.service = self.start_service('pool_manager')
|
||||
with patch.object(
|
||||
central_rpcapi.CentralAPI,
|
||||
'get_pool',
|
||||
return_value=objects.Pool.from_dict(POOL_DICT)):
|
||||
self.service = self.start_service('pool_manager')
|
||||
|
||||
zone = self._build_zone('example.org.', 'UPDATE', 'PENDING')
|
||||
|
||||
|
@ -116,7 +116,7 @@ class StorageTestCase(object):
|
||||
context, pool.id, objects.PoolAlsoNotify.from_dict(values))
|
||||
|
||||
# Paging Tests
|
||||
def _ensure_paging(self, data, method):
|
||||
def _ensure_paging(self, data, method, criterion=None):
|
||||
"""
|
||||
Given an array of created items we iterate through them making sure
|
||||
they match up to things returned by paged results.
|
||||
@ -124,20 +124,27 @@ class StorageTestCase(object):
|
||||
results = None
|
||||
item_number = 0
|
||||
|
||||
criterion = criterion or {}
|
||||
|
||||
for current_page in range(0, int(math.ceil(float(len(data)) / 2))):
|
||||
LOG.debug('Validating results on page %d', current_page)
|
||||
LOG.critical('Validating results on page %d', current_page)
|
||||
|
||||
if results is not None:
|
||||
results = method(
|
||||
self.admin_context, limit=2, marker=results[-1]['id'])
|
||||
self.admin_context,
|
||||
limit=2,
|
||||
marker=results[-1]['id'],
|
||||
criterion=criterion
|
||||
)
|
||||
else:
|
||||
results = method(self.admin_context, limit=2)
|
||||
results = method(self.admin_context, limit=2,
|
||||
criterion=criterion)
|
||||
|
||||
LOG.critical('Results: %d', len(results))
|
||||
|
||||
for result_number, result in enumerate(results):
|
||||
LOG.debug('Validating result %d on page %d', result_number,
|
||||
current_page)
|
||||
LOG.critical('Validating result %d on page %d', result_number,
|
||||
current_page)
|
||||
self.assertEqual(
|
||||
data[item_number]['id'], results[result_number]['id'])
|
||||
|
||||
@ -2402,7 +2409,7 @@ class StorageTestCase(object):
|
||||
criterion = dict(host=pool_nameserver_two['host'])
|
||||
|
||||
results = self.storage.find_pool_nameservers(self.admin_context,
|
||||
criterion)
|
||||
criterion)
|
||||
self.assertEqual(1, len(results))
|
||||
self.assertEqual(pool_nameserver_two['host'], results[0]['host'])
|
||||
|
||||
@ -2535,15 +2542,19 @@ class StorageTestCase(object):
|
||||
def test_find_pool_targets(self):
|
||||
pool = self.create_pool(fixture=0)
|
||||
|
||||
# Verify that there are no pool_targets created
|
||||
actual = self.storage.find_pool_targets(self.admin_context)
|
||||
# Verify that there are no new pool_targets created
|
||||
actual = self.storage.find_pool_targets(
|
||||
self.admin_context,
|
||||
criterion={'pool_id': pool.id})
|
||||
self.assertEqual(0, len(actual))
|
||||
|
||||
# Create a PoolTarget
|
||||
pool_target = self.create_pool_target(pool, fixture=0)
|
||||
|
||||
# Fetch the PoolTargets and ensure only 1 exists
|
||||
actual = self.storage.find_pool_targets(self.admin_context)
|
||||
# Fetch the PoolTargets and ensure only 2 exist
|
||||
actual = self.storage.find_pool_targets(
|
||||
self.admin_context,
|
||||
criterion={'pool_id': pool.id})
|
||||
self.assertEqual(1, len(actual))
|
||||
|
||||
self.assertEqual(pool_target['pool_id'], actual[0]['pool_id'])
|
||||
@ -2557,7 +2568,8 @@ class StorageTestCase(object):
|
||||
for i in range(10)]
|
||||
|
||||
# Ensure we can page through the results.
|
||||
self._ensure_paging(created, self.storage.find_pool_targets)
|
||||
self._ensure_paging(created, self.storage.find_pool_targets,
|
||||
criterion={'pool_id': pool.id})
|
||||
|
||||
def test_find_pool_targets_with_criterion(self):
|
||||
pool = self.create_pool(fixture=0)
|
||||
@ -2582,7 +2594,7 @@ class StorageTestCase(object):
|
||||
criterion = dict(description=pool_target_two['description'])
|
||||
|
||||
results = self.storage.find_pool_targets(self.admin_context,
|
||||
criterion)
|
||||
criterion)
|
||||
self.assertEqual(1, len(results))
|
||||
self.assertEqual(
|
||||
pool_target_two['description'], results[0]['description'])
|
||||
|
156
designate/tests/unit/test_objects/test_yaml_adapters.py
Normal file
156
designate/tests/unit/test_objects/test_yaml_adapters.py
Normal file
@ -0,0 +1,156 @@
|
||||
# Copyright 2014 Hewlett-Packard Development Company, L.P.
|
||||
#
|
||||
# Author: Kiall Mac Innes <kiall@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 os
|
||||
|
||||
import yaml
|
||||
from oslo_log import log as logging
|
||||
import oslotest.base
|
||||
|
||||
from designate import objects
|
||||
from designate.objects import adapters
|
||||
from designate.tests import resources
|
||||
|
||||
|
||||
LOG = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class DesignateYAMLAdapterTest(oslotest.base.BaseTestCase):
|
||||
|
||||
def assertNestedDictContainsSubset(self, expected, actual):
|
||||
for key, value in expected.items():
|
||||
if isinstance(value, dict):
|
||||
self.assertNestedDictContainsSubset(value, actual.get(key, {}))
|
||||
|
||||
elif isinstance(value, list):
|
||||
self.assertEqual(len(value), len(actual[key]))
|
||||
|
||||
for index, item in enumerate(value):
|
||||
self.assertNestedDictContainsSubset(
|
||||
item, actual[key][index])
|
||||
|
||||
else:
|
||||
self.assertEqual(value, actual[key])
|
||||
|
||||
def test_yaml_parsing(self):
|
||||
|
||||
file = os.path.join(resources.path, 'pools_yaml/pools.yaml')
|
||||
with open(file, 'r') as stream:
|
||||
xpools = yaml.safe_load(stream)
|
||||
|
||||
for xpool in xpools:
|
||||
r_pool = adapters.DesignateAdapter.parse(
|
||||
'YAML', xpool, objects.Pool())
|
||||
self.assertEqual('default', r_pool.name)
|
||||
self.assertEqual('Default PowerDNS Pool', r_pool.description)
|
||||
self.assertEqual(2, len(r_pool.ns_records))
|
||||
self.assertEqual(1, r_pool.ns_records[0].priority)
|
||||
self.assertEqual(2, r_pool.ns_records[1].priority)
|
||||
self.assertEqual(
|
||||
'ns1-1.example.org.', r_pool.ns_records[0].hostname)
|
||||
self.assertEqual(
|
||||
'ns1-2.example.org.', r_pool.ns_records[1].hostname)
|
||||
self.assertEqual(1, len(r_pool.targets))
|
||||
self.assertEqual('powerdns', r_pool.targets[0].type)
|
||||
self.assertEqual(
|
||||
'PowerDNS Database Cluster', r_pool.targets[0].description)
|
||||
self.assertEqual(1, len(r_pool.targets[0].masters))
|
||||
self.assertEqual('192.0.2.1', r_pool.targets[0].masters[0].host)
|
||||
self.assertEqual(5354, r_pool.targets[0].masters[0].port)
|
||||
self.assertEqual(1, len(r_pool.targets[0].options))
|
||||
self.assertEqual('connection', r_pool.targets[0].options[0].key)
|
||||
self.assertEqual(
|
||||
'mysql+pymysql://designate:password@127.0.0.1/designate_pdns?charset=utf8', # noqa
|
||||
r_pool.targets[0].options[0].value)
|
||||
self.assertEqual(1, len(r_pool.also_notifies))
|
||||
self.assertEqual('192.0.2.4', r_pool.also_notifies[0].host)
|
||||
self.assertEqual(53, r_pool.also_notifies[0].port)
|
||||
self.assertEqual(1, len(r_pool.attributes))
|
||||
self.assertEqual('type', r_pool.attributes[0].key)
|
||||
self.assertEqual('internal', r_pool.attributes[0].value)
|
||||
|
||||
def test_yaml_rendering(self):
|
||||
|
||||
pool_dict = {
|
||||
'also_notifies': [
|
||||
{
|
||||
'host': u'192.0.2.4',
|
||||
'pool_id': u'cf2e8eab-76cd-4162-bf76-8aeee3556de0',
|
||||
'port': 53,
|
||||
}
|
||||
],
|
||||
'attributes': [],
|
||||
'description': u'Default PowerDNS Pool',
|
||||
'id': u'cf2e8eab-76cd-4162-bf76-8aeee3556de0',
|
||||
'name': u'default',
|
||||
'nameservers': [
|
||||
{
|
||||
'host': u'192.0.2.2',
|
||||
'pool_id': u'cf2e8eab-76cd-4162-bf76-8aeee3556de0',
|
||||
'port': 53,
|
||||
},
|
||||
{
|
||||
'host': u'192.0.2.3',
|
||||
'pool_id': u'cf2e8eab-76cd-4162-bf76-8aeee3556de0',
|
||||
'port': 53,
|
||||
}
|
||||
],
|
||||
'ns_records': [
|
||||
{
|
||||
'hostname': u'ns1-1.example.org.',
|
||||
'pool_id': u'cf2e8eab-76cd-4162-bf76-8aeee3556de0',
|
||||
'priority': 1,
|
||||
},
|
||||
{
|
||||
'hostname': u'ns1-2.example.org.',
|
||||
'pool_id': u'cf2e8eab-76cd-4162-bf76-8aeee3556de0',
|
||||
'priority': 2,
|
||||
}
|
||||
],
|
||||
'provisioner': u'UNMANAGED',
|
||||
'targets': [
|
||||
{
|
||||
'description': u'PowerDNS Database Cluster',
|
||||
'masters': [
|
||||
{
|
||||
'host': u'192.0.2.1',
|
||||
'pool_target_id': u'd567d569-2d69-41d5-828d-f7054bb10b5c', # noqa
|
||||
'port': 5354,
|
||||
}
|
||||
],
|
||||
'options': [
|
||||
{
|
||||
'key': u'connection',
|
||||
'pool_target_id': u'd567d569-2d69-41d5-828d-f7054bb10b5c', # noqa
|
||||
'value': u'mysql+pymysql://designate:password@127.0.0.1/designate_pdns?charset=utf8', # noqa
|
||||
}
|
||||
],
|
||||
'pool_id': u'cf2e8eab-76cd-4162-bf76-8aeee3556de0',
|
||||
'type': u'powerdns',
|
||||
}
|
||||
]
|
||||
}
|
||||
|
||||
file = os.path.join(resources.path, 'pools_yaml/sample_output.yaml')
|
||||
with open(file, 'r') as stream:
|
||||
self.assertEqual(
|
||||
stream.read(),
|
||||
yaml.dump(
|
||||
adapters.DesignateAdapter.render(
|
||||
'YAML', objects.PoolList.from_list([pool_dict])
|
||||
),
|
||||
default_flow_style=False
|
||||
)
|
||||
)
|
@ -33,6 +33,55 @@ from designate.tests.unit import RwObject
|
||||
import designate.pool_manager.service as pm_module
|
||||
|
||||
|
||||
POOL_DICT = {
|
||||
'also_notifies': [
|
||||
{
|
||||
'host': u'192.0.2.4',
|
||||
'pool_id': u'cf2e8eab-76cd-4162-bf76-8aeee3556de0',
|
||||
'port': 53,
|
||||
}
|
||||
],
|
||||
'attributes': [],
|
||||
'description': u'Default PowerDNS Pool',
|
||||
'id': u'cf2e8eab-76cd-4162-bf76-8aeee3556de0',
|
||||
'name': u'default',
|
||||
'nameservers': [
|
||||
{
|
||||
'host': u'192.0.2.2',
|
||||
'pool_id': u'cf2e8eab-76cd-4162-bf76-8aeee3556de0',
|
||||
'port': 53,
|
||||
},
|
||||
{
|
||||
'host': u'192.0.2.3',
|
||||
'pool_id': u'cf2e8eab-76cd-4162-bf76-8aeee3556de0',
|
||||
'port': 53,
|
||||
}
|
||||
],
|
||||
'ns_records': [
|
||||
{
|
||||
'hostname': u'ns1-1.example.org.',
|
||||
'pool_id': u'cf2e8eab-76cd-4162-bf76-8aeee3556de0',
|
||||
'priority': 1,
|
||||
},
|
||||
{
|
||||
'hostname': u'ns1-2.example.org.',
|
||||
'pool_id': u'cf2e8eab-76cd-4162-bf76-8aeee3556de0',
|
||||
'priority': 2,
|
||||
}
|
||||
],
|
||||
'provisioner': u'UNMANAGED',
|
||||
'targets': [
|
||||
{
|
||||
'description': u'PowerDNS Database Cluster',
|
||||
'masters': [],
|
||||
'options': [],
|
||||
'type': 'fake',
|
||||
'pool_id': u'cf2e8eab-76cd-4162-bf76-8aeee3556de0',
|
||||
}
|
||||
]
|
||||
}
|
||||
|
||||
|
||||
class PoolManagerInitTest(test.BaseTestCase):
|
||||
def __setUp(self):
|
||||
super(PoolManagerTest, self).setUp()
|
||||
@ -44,21 +93,20 @@ class PoolManagerInitTest(test.BaseTestCase):
|
||||
self.assertRaises(exceptions.NoPoolTargetsConfigured, Service)
|
||||
|
||||
def test_init(self):
|
||||
with patch.object(objects.Pool, 'from_config',
|
||||
return_value=Mock()):
|
||||
Service._setup_target_backends = Mock()
|
||||
Service()
|
||||
|
||||
def test_start(self):
|
||||
with patch.object(objects.Pool, 'from_config',
|
||||
return_value=Mock()):
|
||||
Service._setup_target_backends = Mock()
|
||||
pm = Service()
|
||||
pm.pool.targets = ()
|
||||
pm.tg.add_timer = Mock()
|
||||
pm._pool_election = Mock()
|
||||
with patch("designate.service.RPCService.start"):
|
||||
pm.start()
|
||||
with patch.object(
|
||||
pm.central_api,
|
||||
'get_pool',
|
||||
return_value=objects.Pool.from_dict(POOL_DICT)):
|
||||
pm.start()
|
||||
|
||||
call1 = pm.tg.add_timer.call_args_list[0][0]
|
||||
self.assertEqual(120, call1[0])
|
||||
@ -88,8 +136,11 @@ class PoolManagerTest(test.BaseTestCase):
|
||||
def setUp(self, *mocks):
|
||||
super(PoolManagerTest, self).setUp()
|
||||
self.pm = Service()
|
||||
self.pm.pool.targets = ()
|
||||
self.pm.tg.add_timer = Mock()
|
||||
self.pm.pool = Mock()
|
||||
setattr(self.pm.pool, 'targets', ())
|
||||
setattr(self.pm.pool, 'also_notifies', ())
|
||||
setattr(self.pm.pool, 'nameservers', ())
|
||||
self.pm._pool_election = Mock()
|
||||
self.pm.target_backends = {}
|
||||
|
||||
|
@ -29,7 +29,7 @@ DESIGNATE_AKAMAI_USERNAME=${DESIGNATE_AKAMAI_USERNAME:-username}
|
||||
DESIGNATE_AKAMAI_PASSWORD=${DESIGNATE_AKAMAI_PASSWORD:-password}
|
||||
DESIGNATE_AKAMAI_MASTERS=${DESIGNATE_AKAMAI_MASTERS:-"$DESIGNATE_SERVICE_HOST:$DESIGNATE_SERVICE_PORT_MDNS"}
|
||||
DESIGNATE_AKAMAI_NAMESERVERS=${DESIGNATE_AKAMAI_NAMESERVERS:-""}
|
||||
DESIGNATE_AKAMAI_ALSO_NOTIFIES=${DESIGNATE_AKAMAI_ALSO_NOTIFIES:-"193.108.155.34:53,23.73.134.141:53,80.67.64.148:53,23.73.134.237:53,23.73.133.141:53,23.73.133.237:53,80.67.64.10:53,72.246.0.10:53,72.247.45.157:53,72.246.192.168:53,193.108.152.143:53,60.254.128.45:53,72.247.45.110:53,72.247.45.65:53,72.247.45.25:53"}
|
||||
DESIGNATE_AKAMAI_ALSO_NOTIFIES=${DESIGNATE_AKAMAI_ALSO_NOTIFIES:-"193.108.155.34,23.73.134.141,80.67.64.148,23.73.134.237,23.73.133.141,23.73.133.237,80.67.64.10,72.246.0.10,72.247.45.157,72.246.192.168,193.108.152.143,60.254.128.45,72.247.45.110,72.247.45.65,72.247.45.25"}
|
||||
|
||||
# Pull in DESIGNATE_3RDPARTY_CREDS user/pass if set
|
||||
if [ -n "$DESIGNATE_3RDPARTY_CREDS" ]; then
|
||||
@ -57,38 +57,72 @@ function install_designate_backend {
|
||||
|
||||
# configure_designate_backend - make configuration changes, including those to other services
|
||||
function configure_designate_backend {
|
||||
iniset $DESIGNATE_CONF pool_target:$DESIGNATE_TARGET_ID type akamai
|
||||
iniset $DESIGNATE_CONF pool_target:$DESIGNATE_TARGET_ID masters $DESIGNATE_AKAMAI_MASTERS
|
||||
iniset $DESIGNATE_CONF pool_target:$DESIGNATE_TARGET_ID options "username: $DESIGNATE_AKAMAI_USERNAME, password: $DESIGNATE_AKAMAI_PASSWORD"
|
||||
# Generate Designate pool.yaml file
|
||||
sudo tee $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
---
|
||||
- name: default
|
||||
description: DevStack Akamai Pool
|
||||
attributes: {}
|
||||
|
||||
# Create a Pool Nameserver for each of the Akamai nameservers
|
||||
local nameserver_ids=""
|
||||
IFS=',' read -a nameservers <<< "$DESIGNATE_AKAMAI_NAMESERVERS"
|
||||
targets:
|
||||
- type: akamai
|
||||
description: Akamai API
|
||||
|
||||
for nameserver in "${nameservers[@]}"; do
|
||||
local nameserver_id=`uuidgen`
|
||||
iniset $DESIGNATE_CONF pool_nameserver:$nameserver_id host $(dig +short A $nameserver | head -n 1)
|
||||
iniset $DESIGNATE_CONF pool_nameserver:$nameserver_id port 53
|
||||
options:
|
||||
username: $DESIGNATE_AKAMAI_USERNAME
|
||||
password: $DESIGNATE_AKAMAI_PASSWORD
|
||||
|
||||
# Append the Nameserver ID to the list
|
||||
nameserver_ids+=${nameserver_id},
|
||||
masters:
|
||||
EOF
|
||||
|
||||
# Create a Pool Master for each of the Akamai Masters
|
||||
IFS=',' read -a masters <<< "$DESIGNATE_AKAMAI_MASTERS"
|
||||
|
||||
for master in "${masters[@]}"; do
|
||||
sudo tee -a $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
- host: $master
|
||||
port: 53
|
||||
EOF
|
||||
done
|
||||
|
||||
# Configure the Pool for the set of nameserver IDs, minus the trailing comma
|
||||
iniset $DESIGNATE_CONF pool:$DESIGNATE_POOL_ID nameservers "${nameserver_ids:0:-1}"
|
||||
# Create a Pool NS Record for each of the Akamai Nameservers
|
||||
IFS=',' read -a nameservers <<< "$DESIGNATE_AKAMAI_NAMESERVERS"
|
||||
|
||||
# Configure the Pool to Notify Akamai's Transfer Agents
|
||||
iniset $DESIGNATE_CONF pool:$DESIGNATE_POOL_ID also_notifies "$DESIGNATE_AKAMAI_ALSO_NOTIFIES"
|
||||
}
|
||||
sudo tee -a $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
ns_records:
|
||||
EOF
|
||||
|
||||
# create_designate_ns_records - Create Pool NS Records
|
||||
function create_designate_ns_records_backend {
|
||||
# Build an array of the Akamai nameservers.
|
||||
IFS=',' read -a ns_records <<< "$DESIGNATE_AKAMAI_NAMESERVERS"
|
||||
for nameserver in "${nameservers[@]}"; do
|
||||
sudo tee -a $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
- hostname: $nameserver
|
||||
priority: 1
|
||||
EOF
|
||||
done
|
||||
|
||||
# Create a NS Record for each of the Akamai nameservers
|
||||
for ns_record in "${ns_records[@]}"; do
|
||||
designate server-create --name "${ns_record%%.}."
|
||||
# Create a Pool Nameserver for each of the Akamai Nameservers
|
||||
sudo tee -a $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
nameservers:
|
||||
EOF
|
||||
|
||||
for nameserver in "${nameservers[@]}"; do
|
||||
sudo tee -a $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
- host: `dig +short A $nameserver | head -n 1`
|
||||
port: 53
|
||||
EOF
|
||||
done
|
||||
|
||||
# Create a Pool Also Notifies for each of the Akamai Also Notifies
|
||||
IFS=',' read -a also_notifies <<< "$DESIGNATE_AKAMAI_ALSO_NOTIFIES"
|
||||
|
||||
sudo tee -a $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
also_notifies:
|
||||
EOF
|
||||
|
||||
for also_notify in "${also_notifies[@]}"; do
|
||||
sudo tee -a $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
- host: $also_notify
|
||||
port: 53
|
||||
EOF
|
||||
done
|
||||
}
|
||||
|
||||
|
@ -74,15 +74,37 @@ EOF
|
||||
|
||||
# configure_designate_backend - make configuration changes, including those to other services
|
||||
function configure_designate_backend {
|
||||
iniset $DESIGNATE_CONF pool_target:$DESIGNATE_TARGET_ID type bind9
|
||||
iniset $DESIGNATE_CONF pool_target:$DESIGNATE_TARGET_ID masters $DESIGNATE_SERVICE_HOST:$DESIGNATE_SERVICE_PORT_MDNS
|
||||
iniset $DESIGNATE_CONF pool_target:$DESIGNATE_TARGET_ID options "rndc_host: $DESIGNATE_SERVICE_HOST, rndc_port: $DESIGNATE_SERVICE_PORT_RNDC, rndc_config_file: $BIND_CFG_DIR/rndc.conf, rndc_key_file: $BIND_CFG_DIR/rndc.key, host: $DESIGNATE_SERVICE_HOST, port: $DESIGNATE_SERVICE_PORT_DNS"
|
||||
# Generate Designate pool.yaml file
|
||||
sudo tee $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
---
|
||||
- name: default
|
||||
description: DevStack BIND Pool
|
||||
attributes: {}
|
||||
|
||||
# DevStack Managed BIND NameServer
|
||||
local nameserver_id=`uuidgen`
|
||||
iniset $DESIGNATE_CONF pool:$DESIGNATE_POOL_ID nameservers $nameserver_id
|
||||
iniset $DESIGNATE_CONF pool_nameserver:$nameserver_id host $DESIGNATE_SERVICE_HOST
|
||||
iniset $DESIGNATE_CONF pool_nameserver:$nameserver_id port $DESIGNATE_SERVICE_PORT_DNS
|
||||
ns_records:
|
||||
- hostname: $DESIGNATE_DEFAULT_NS_RECORD
|
||||
priority: 1
|
||||
|
||||
nameservers:
|
||||
- host: $DESIGNATE_SERVICE_HOST
|
||||
port: $DESIGNATE_SERVICE_PORT_DNS
|
||||
|
||||
targets:
|
||||
- type: bind9
|
||||
description: BIND Instance
|
||||
|
||||
masters:
|
||||
- host: $DESIGNATE_SERVICE_HOST
|
||||
port: $DESIGNATE_SERVICE_PORT_MDNS
|
||||
|
||||
options:
|
||||
host: $DESIGNATE_SERVICE_HOST
|
||||
port: $DESIGNATE_SERVICE_PORT_DNS
|
||||
rndc_host: $DESIGNATE_SERVICE_HOST
|
||||
rndc_port: $DESIGNATE_SERVICE_PORT_RNDC
|
||||
rndc_config_file: $BIND_CFG_DIR/rndc.conf
|
||||
rndc_key_file: $BIND_CFG_DIR/rndc.key
|
||||
EOF
|
||||
|
||||
sudo chown $STACK_USER $BIND_CFG_DIR
|
||||
|
||||
|
@ -33,7 +33,7 @@ DESIGNATE_DYNECT_JOB_TIMEOUT=${DESIGNATE_DYNECT_JOB_TIMEOUT:-}
|
||||
DESIGNATE_DYNECT_TIMEOUT=${DESIGNATE_DYNECT_TIMEOUT:-}
|
||||
DESIGNATE_DYNECT_MASTERS=${DESIGNATE_DYNECT_MASTERS:-"$DESIGNATE_SERVICE_HOST:$DESIGNATE_SERVICE_PORT_MDNS"}
|
||||
DESIGNATE_DYNECT_NAMESERVERS=${DESIGNATE_DYNECT_NAMESERVERS:-""}
|
||||
DESIGNATE_DYNECT_ALSO_NOTIFIES=${DESIGNATE_DYNECT_ALSO_NOTIFIES:-"204.13.249.65:53,208.78.68.65:53"}
|
||||
DESIGNATE_DYNECT_ALSO_NOTIFIES=${DESIGNATE_DYNECT_ALSO_NOTIFIES:-"204.13.249.65,208.78.68.65"}
|
||||
|
||||
# Pull in DESIGNATE_3RDPARTY_CREDS user/pass if set
|
||||
if [ -n "$DESIGNATE_3RDPARTY_CREDS" ]; then
|
||||
@ -62,47 +62,73 @@ function install_designate_backend {
|
||||
|
||||
# configure_designate_backend - make configuration changes, including those to other services
|
||||
function configure_designate_backend {
|
||||
iniset $DESIGNATE_CONF pool_target:$DESIGNATE_TARGET_ID type dynect
|
||||
iniset $DESIGNATE_CONF pool_target:$DESIGNATE_TARGET_ID masters $DESIGNATE_DYNECT_MASTERS
|
||||
iniset $DESIGNATE_CONF pool_target:$DESIGNATE_TARGET_ID options "customer_name: $DESIGNATE_DYNECT_CUSTOMER, username: $DESIGNATE_DYNECT_USERNAME, password: $DESIGNATE_DYNECT_PASSWORD"
|
||||
# Generate Designate pool.yaml file
|
||||
sudo tee $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
---
|
||||
- name: default
|
||||
description: DevStack DynECT Pool
|
||||
attributes: {}
|
||||
|
||||
# Create a Pool Nameserver for each of the DynECT nameservers
|
||||
local nameserver_ids=""
|
||||
IFS=',' read -a nameservers <<< "$DESIGNATE_DYNECT_NAMESERVERS"
|
||||
targets:
|
||||
- type: dynect
|
||||
description: DynECT API
|
||||
|
||||
for nameserver in "${nameservers[@]}"; do
|
||||
local nameserver_id=`uuidgen`
|
||||
iniset $DESIGNATE_CONF pool_nameserver:$nameserver_id host $(dig +short A $nameserver | head -n 1)
|
||||
iniset $DESIGNATE_CONF pool_nameserver:$nameserver_id port 53
|
||||
options:
|
||||
customer_name: $DESIGNATE_DYNECT_CUSTOMER
|
||||
username: $DESIGNATE_DYNECT_USERNAME
|
||||
password: $DESIGNATE_DYNECT_PASSWORD
|
||||
|
||||
# Append the Nameserver ID to the list
|
||||
nameserver_ids+=${nameserver_id},
|
||||
masters:
|
||||
EOF
|
||||
|
||||
# Create a Pool Master for each of the DynECT Masters
|
||||
IFS=',' read -a masters <<< "$DESIGNATE_DYNECT_MASTERS"
|
||||
|
||||
for master in "${masters[@]}"; do
|
||||
sudo tee -a $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
- host: $master
|
||||
port: 53
|
||||
EOF
|
||||
done
|
||||
|
||||
# Configure the Pool for the set of nameserver IDs, minus the trailing comma
|
||||
iniset $DESIGNATE_CONF pool:$DESIGNATE_POOL_ID nameservers "${nameserver_ids:0:-1}"
|
||||
# Create a Pool NS Record for each of the DynECT Nameservers
|
||||
IFS=',' read -a nameservers <<< "$DESIGNATE_DYNECT_NAMESERVERS"
|
||||
|
||||
# Configure the Pool to Notify DynECT's Transfer Agents
|
||||
iniset $DESIGNATE_CONF pool:$DESIGNATE_POOL_ID also_notifies "$DESIGNATE_DYNECT_ALSO_NOTIFIES"
|
||||
sudo tee -a $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
ns_records:
|
||||
EOF
|
||||
|
||||
# Global DynECT Backend Settings
|
||||
if [ ! -z $DESIGNATE_DYNECT_JOB_TIMEOUT ]; then
|
||||
iniset $DESIGNATE_CONF backend:dynect job_timeout "$DESIGNATE_DYNECT_JOB_TIMEOUT"
|
||||
fi
|
||||
for nameserver in "${nameservers[@]}"; do
|
||||
sudo tee -a $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
- hostname: $nameserver
|
||||
priority: 1
|
||||
EOF
|
||||
done
|
||||
|
||||
if [ ! -z $DESIGNATE_DYNECT_TIMEOUT ]; then
|
||||
iniset $DESIGNATE_CONF backend:dynect timeout "$DESIGNATE_DYNECT_TIMEOUT"
|
||||
fi
|
||||
}
|
||||
# Create a Pool Nameserver for each of the DynECT Nameservers
|
||||
sudo tee -a $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
nameservers:
|
||||
EOF
|
||||
|
||||
# create_designate_ns_records - Create Pool NS Records
|
||||
function create_designate_ns_records_backend {
|
||||
# Build an array of the DynECT nameservers.
|
||||
IFS=',' read -a ns_records <<< "$DESIGNATE_DYNECT_NAMESERVERS"
|
||||
for nameserver in "${nameservers[@]}"; do
|
||||
sudo tee -a $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
- host: `dig +short A $nameserver | head -n 1`
|
||||
port: 53
|
||||
EOF
|
||||
done
|
||||
|
||||
# Create a NS Record for each of the DynECT nameservers
|
||||
for ns_record in "${ns_records[@]}"; do
|
||||
designate server-create --name "${ns_record%%.}."
|
||||
# Create a Pool Also Notifies for each of the DynECT Also Notifies
|
||||
IFS=',' read -a also_notifies <<< "$DESIGNATE_DYNECT_ALSO_NOTIFIES"
|
||||
|
||||
sudo tee -a $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
also_notifies:
|
||||
EOF
|
||||
|
||||
for also_notify in "${also_notifies[@]}"; do
|
||||
sudo tee -a $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
- host: $also_notify
|
||||
port: 53
|
||||
EOF
|
||||
done
|
||||
}
|
||||
|
||||
|
@ -31,7 +31,21 @@ function install_designate_backend {
|
||||
|
||||
# configure_designate_backend - make configuration changes, including those to other services
|
||||
function configure_designate_backend {
|
||||
:
|
||||
# Generate Designate pool.yaml file
|
||||
sudo tee $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
---
|
||||
- name: default
|
||||
description: DevStack Fake Pool
|
||||
attributes: {}
|
||||
|
||||
ns_records:
|
||||
- hostname: $DESIGNATE_DEFAULT_NS_RECORD
|
||||
priority: 1
|
||||
|
||||
targets:
|
||||
- type: fake
|
||||
description: Fake Backend
|
||||
EOF
|
||||
}
|
||||
|
||||
# init_designate_backend - initialize databases, etc.
|
||||
|
@ -53,15 +53,6 @@ function install_designate_backend {
|
||||
|
||||
# configure_designate_backend - make configuration changes, including those to other services
|
||||
function configure_designate_backend {
|
||||
iniset $DESIGNATE_CONF pool_target:$DESIGNATE_TARGET_ID type infoblox
|
||||
iniset $DESIGNATE_CONF pool_target:$DESIGNATE_TARGET_ID masters $DESIGNATE_SERVICE_HOST:$DESIGNATE_SERVICE_PORT_MDNS
|
||||
iniset $DESIGNATE_CONF pool_target:$DESIGNATE_TARGET_ID options "host: $DESIGNATE_INFOBLOX_NAMESERVER_IP, port: $DESIGNATE_SERVICE_PORT_DNS"
|
||||
|
||||
local nameserver_id=`uuidgen`
|
||||
iniset $DESIGNATE_CONF pool:$DESIGNATE_POOL_ID nameservers $nameserver_id
|
||||
iniset $DESIGNATE_CONF pool_nameserver:$nameserver_id host $DESIGNATE_INFOBLOX_NAMESERVER_IP
|
||||
iniset $DESIGNATE_CONF pool_nameserver:$nameserver_id port $DESIGNATE_SERVICE_PORT_DNS
|
||||
|
||||
iniset $DESIGNATE_CONF backend:infoblox wapi_url $DESIGNATE_INFOBLOX_WAPI_URL
|
||||
iniset $DESIGNATE_CONF backend:infoblox username $DESIGNATE_INFOBLOX_USERNAME
|
||||
iniset $DESIGNATE_CONF backend:infoblox password $DESIGNATE_INFOBLOX_PASSWORD
|
||||
@ -72,6 +63,34 @@ function configure_designate_backend {
|
||||
iniset $DESIGNATE_CONF backend:infoblox dns_view $DESIGNATE_INFOBLOX_DNS_VIEW
|
||||
iniset $DESIGNATE_CONF backend:infoblox net_view $DESIGNATE_INFOBLOX_NET_VIEW
|
||||
iniset $DESIGNATE_CONF backend:infoblox ns_group $DESIGNATE_INFOBLOX_NS_GROUP
|
||||
|
||||
# Generate Designate pool.yaml file
|
||||
sudo tee $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
---
|
||||
- name: default
|
||||
description: DevStack InfoBlox Pool
|
||||
attributes: {}
|
||||
|
||||
ns_records:
|
||||
- hostname: $DESIGNATE_DEFAULT_NS_RECORD
|
||||
priority: 1
|
||||
|
||||
nameservers:
|
||||
- host: $DESIGNATE_INFOBLOX_NAMESERVER_IP
|
||||
port: $DESIGNATE_SERVICE_PORT_DNS
|
||||
|
||||
targets:
|
||||
- type: powerdns
|
||||
description: InfoBlox Cluster API
|
||||
|
||||
masters:
|
||||
- host: $DESIGNATE_SERVICE_HOST
|
||||
port: $DESIGNATE_SERVICE_PORT_MDNS
|
||||
|
||||
options:
|
||||
host: $DESIGNATE_INFOBLOX_NAMESERVER_IP
|
||||
port: $DESIGNATE_SERVICE_PORT_DNS
|
||||
EOF
|
||||
}
|
||||
|
||||
# init_designate_backend - initialize databases, etc.
|
||||
|
@ -1,77 +0,0 @@
|
||||
# Configure the ipa backend
|
||||
|
||||
# Enable with:
|
||||
# DESIGNATE_BACKEND_DRIVER=ipa
|
||||
|
||||
# Dependencies:
|
||||
# ``functions`` file
|
||||
# ``designate`` configuration
|
||||
|
||||
# install_designate_backend - install any external requirements
|
||||
# configure_designate_backend - make configuration changes, including those to other services
|
||||
# init_designate_backend - initialize databases, etc.
|
||||
# start_designate_backend - start any external services
|
||||
# stop_designate_backend - stop any external services
|
||||
# cleanup_designate_backend - remove transient data and cache
|
||||
|
||||
# Save trace setting
|
||||
DP_IPA_XTRACE=$(set +o | grep xtrace)
|
||||
set +o xtrace
|
||||
|
||||
# Defaults
|
||||
# --------
|
||||
DESIGNATE_IPA_HOST=${DESIGNATE_IPA_HOST:-$(hostname)}
|
||||
DESIGNATE_IPA_CA_CERT=${DESIGNATE_IPA_CA_CERT:-/etc/ipa/ca.crt}
|
||||
DESIGNATE_IPA_CLIENT_KEYTAB=${DESIGNATE_IPA_CLIENT_KEYTAB:-$HOME/ipaadmin.keytab}
|
||||
|
||||
# Entry Points
|
||||
# ------------
|
||||
|
||||
# install_designate_backend - install any external requirements
|
||||
function install_designate_backend {
|
||||
:
|
||||
}
|
||||
|
||||
# configure_designate_backend - make configuration changes, including those to other services
|
||||
function configure_designate_backend {
|
||||
rc=0
|
||||
if [[ ! -f $DESIGNATE_IPA_CA_CERT ]]; then
|
||||
echo ERROR: IPA CA cert file $DESIGNATE_IPA_CA_CERT not found
|
||||
rc=1
|
||||
fi
|
||||
if [[ ! -f $DESIGNATE_IPA_CLIENT_KEYTAB ]]; then
|
||||
echo ERROR: IPA client keytab file $DESIGNATE_IPA_CLIENT_KEYTAB not found
|
||||
rc=1
|
||||
fi
|
||||
if [[ $rc = 1 ]]; then
|
||||
die $LINENO "Error with IPA configuration"
|
||||
fi
|
||||
iniset $DESIGNATE_CONF backend:ipa ipa_host $DESIGNATE_IPA_HOST
|
||||
iniset $DESIGNATE_CONF backend:ipa ipa_ca_cert $DESIGNATE_IPA_CA_CERT
|
||||
iniset $DESIGNATE_CONF backend:ipa ipa_client_keytab $DESIGNATE_IPA_CLIENT_KEYTAB
|
||||
# devstack tests use dummy NS records, so tell IPA to allow this
|
||||
iniset $DESIGNATE_CONF backend:ipa ipa_force_ns_use True
|
||||
}
|
||||
|
||||
# init_designate_backend - initialize databases, etc.
|
||||
function init_designate_backend {
|
||||
:
|
||||
}
|
||||
|
||||
# start_designate_backend - start any external services
|
||||
function start_designate_backend {
|
||||
:
|
||||
}
|
||||
|
||||
# stop_designate_backend - stop any external services
|
||||
function stop_designate_backend {
|
||||
:
|
||||
}
|
||||
|
||||
# cleanup_designate_backend - remove transient data and cache
|
||||
function cleanup_designate_backend {
|
||||
:
|
||||
}
|
||||
|
||||
# Restore xtrace
|
||||
$DP_IPA_XTRACE
|
@ -53,16 +53,36 @@ function install_designate_backend {
|
||||
|
||||
# configure_designate_backend - make configuration changes, including those to other services
|
||||
function configure_designate_backend {
|
||||
iniset $DESIGNATE_CONF pool_target:$DESIGNATE_TARGET_ID type powerdns
|
||||
iniset $DESIGNATE_CONF pool_target:$DESIGNATE_TARGET_ID masters $DESIGNATE_SERVICE_HOST:$DESIGNATE_SERVICE_PORT_MDNS
|
||||
iniset $DESIGNATE_CONF pool_target:$DESIGNATE_TARGET_ID options "connection: `database_connection_url designate_pdns`, host: $DESIGNATE_SERVICE_HOST, port: $DESIGNATE_SERVICE_PORT_DNS"
|
||||
# Generate Designate pool.yaml file
|
||||
sudo tee $DESIGNATE_CONF_DIR/pools.yaml > /dev/null <<EOF
|
||||
---
|
||||
- name: default
|
||||
description: DevStack PowerDNS Pool
|
||||
attributes: {}
|
||||
|
||||
# DevStack Managed PDNS NameServer
|
||||
local nameserver_id=`uuidgen`
|
||||
iniset $DESIGNATE_CONF pool:$DESIGNATE_POOL_ID nameservers $nameserver_id
|
||||
iniset $DESIGNATE_CONF pool_nameserver:$nameserver_id host $DESIGNATE_SERVICE_HOST
|
||||
iniset $DESIGNATE_CONF pool_nameserver:$nameserver_id port $DESIGNATE_SERVICE_PORT_DNS
|
||||
ns_records:
|
||||
- hostname: $DESIGNATE_DEFAULT_NS_RECORD
|
||||
priority: 1
|
||||
|
||||
nameservers:
|
||||
- host: $DESIGNATE_SERVICE_HOST
|
||||
port: $DESIGNATE_SERVICE_PORT_DNS
|
||||
|
||||
targets:
|
||||
- type: powerdns
|
||||
description: PowerDNS Database Cluster
|
||||
|
||||
masters:
|
||||
- host: $DESIGNATE_SERVICE_HOST
|
||||
port: $DESIGNATE_SERVICE_PORT_MDNS
|
||||
|
||||
options:
|
||||
host: $DESIGNATE_SERVICE_HOST
|
||||
port: $DESIGNATE_SERVICE_PORT_DNS
|
||||
connection: '`database_connection_url designate_pdns`'
|
||||
EOF
|
||||
|
||||
# Generate PowerDNS pdns.conf file
|
||||
sudo tee $POWERDNS_CFG_DIR/pdns.conf > /dev/null <<EOF
|
||||
# General Config
|
||||
setgid=pdns
|
||||
@ -123,9 +143,12 @@ function init_designate_backend {
|
||||
|
||||
# (Re)create designate_pdns database
|
||||
recreate_database designate_pdns utf8
|
||||
}
|
||||
|
||||
# create_designate_pool_configuration_backend - Perform post-pool config tasks
|
||||
function create_designate_pool_configuration_backend {
|
||||
# Init and migrate designate_pdns database
|
||||
designate-manage powerdns sync $DESIGNATE_TARGET_ID
|
||||
designate-manage powerdns sync $DESIGNATE_POOL_ID
|
||||
}
|
||||
|
||||
# start_designate_backend - start any external services
|
||||
@ -133,6 +156,7 @@ function start_designate_backend {
|
||||
start_service pdns
|
||||
}
|
||||
|
||||
|
||||
# stop_designate_backend - stop any external services
|
||||
function stop_designate_backend {
|
||||
stop_service pdns
|
||||
|
@ -49,6 +49,7 @@ function configure_designate {
|
||||
|
||||
# General Configuration
|
||||
iniset_rpc_backend designate $DESIGNATE_CONF DEFAULT
|
||||
iniset $DESIGNATE_CONF DEFAULT rpc_response_timeout 5
|
||||
|
||||
iniset $DESIGNATE_CONF DEFAULT debug $ENABLE_DEBUG_LOG_LEVEL
|
||||
iniset $DESIGNATE_CONF DEFAULT verbose True
|
||||
@ -76,9 +77,6 @@ function configure_designate {
|
||||
iniset $DESIGNATE_CONF pool_manager_cache:sqlalchemy connection `database_connection_url designate_pool_manager`
|
||||
fi
|
||||
|
||||
# Pool Options
|
||||
iniset $DESIGNATE_CONF pool:$DESIGNATE_POOL_ID targets $DESIGNATE_TARGET_ID
|
||||
|
||||
# API Configuration
|
||||
sudo cp $DESIGNATE_DIR/etc/designate/api-paste.ini $DESIGNATE_APIPASTE_CONF
|
||||
iniset $DESIGNATE_CONF service:api enabled_extensions_v1 $DESIGNATE_ENABLED_EXTENSIONS_V1
|
||||
@ -185,13 +183,14 @@ function create_designate_accounts {
|
||||
fi
|
||||
}
|
||||
|
||||
# create_designate_ns_records - Create Pool NS Records
|
||||
function create_designate_ns_records {
|
||||
# Allow Backends to install their own NS Records rather than the default
|
||||
if function_exists create_designate_ns_records_backend; then
|
||||
create_designate_ns_records_backend
|
||||
else
|
||||
designate server-create --name $DESIGNATE_DEFAULT_NS_RECORD
|
||||
# create_designate_pool_configuration - Create Pool Configuration
|
||||
function create_designate_pool_configuration {
|
||||
# Sync Pools Config
|
||||
designate-manage pool update --file $DESIGNATE_CONF_DIR/pools.yaml
|
||||
|
||||
# Allow Backends to do backend specific tasks
|
||||
if function_exists create_designate_pool_configuration_backend; then
|
||||
create_designate_pool_configuration_backend
|
||||
fi
|
||||
}
|
||||
|
||||
@ -320,14 +319,14 @@ if is_service_enabled designate; then
|
||||
echo_summary "Initializing Designate"
|
||||
init_designate
|
||||
|
||||
echo "Configuring Tempest options for Designate"
|
||||
echo_summary "Configuring Tempest options for Designate"
|
||||
configure_designate_tempest
|
||||
|
||||
echo_summary "Starting Designate"
|
||||
start_designate
|
||||
|
||||
echo_summary "Creating pool NS records"
|
||||
create_designate_ns_records
|
||||
echo_summary "Creating Pool Configuration"
|
||||
create_designate_pool_configuration
|
||||
fi
|
||||
|
||||
if [[ "$1" == "unstack" ]]; then
|
||||
|
@ -2,7 +2,6 @@
|
||||
DESIGNATE_BACKEND_DRIVER=${DESIGNATE_BACKEND_DRIVER:=powerdns}
|
||||
DESIGNATE_POOL_MANAGER_CACHE_DRIVER=${DESIGNATE_POOL_MANAGER_CACHE_DRIVER:-memcache}
|
||||
DESIGNATE_POOL_ID=${DESIGNATE_POOL_ID:-794ccc2c-d751-44fe-b57f-8894c9f5c842}
|
||||
DESIGNATE_TARGET_ID=${DESIGNATE_TARGET_ID:-f26e0b32-736f-4f0a-831b-039a415c481e}
|
||||
DESIGNATE_DEFAULT_NS_RECORD=${DESIGNATE_DEFAULT_NS_RECORD:-ns1.devstack.org.}
|
||||
DESIGNATE_NOTIFICATION_DRIVER=${DESIGNATE_NOTIFICATION_DRIVER:-}
|
||||
DESIGNATE_NOTIFICATION_TOPICS=${DESIGNATE_NOTIFICATION_TOPICS:-notifications}
|
||||
|
@ -62,3 +62,64 @@ but it will **not** trigger zones to be moved from one pool to another.
|
||||
As more filters are merged there will be support for dynamic filters.
|
||||
|
||||
|
||||
Managing Pools
|
||||
==============
|
||||
|
||||
In mitaka we moved the method of updating pools to a CLI in `designate-manage`
|
||||
|
||||
There is a YAML file that defines the pool, and is used to load this information into the database.
|
||||
|
||||
|
||||
.. literalinclude:: ../../etc/designate/pools.yaml.sample
|
||||
:language: yaml
|
||||
|
||||
Designate Manage Pools Command Reference
|
||||
----------------------------------------
|
||||
|
||||
Update Pools Information
|
||||
^^^^^^^^^^^^^^^^^^^^^^^^
|
||||
|
||||
.. code-block:: language
|
||||
|
||||
designate-manage pool update [options]
|
||||
|
||||
Options:
|
||||
""""""""
|
||||
|
||||
--file Input file (Default: ``/etc/designate/pools.yaml``)
|
||||
--dry_run Simulate an update. (Default: False)
|
||||
--delete Delete Pools that are not in the input file (Defaults: False)
|
||||
|
||||
.. warning::
|
||||
|
||||
| Running with ``--delete True`` can be **extremely** dangerous.
|
||||
| It will delete any pools that are not in the supplied YAML file, and any
|
||||
| zones that are in that Pool.
|
||||
| Before running with ``--delete True`` we recommend operators run with
|
||||
| ``--delete True --dry_run True`` to view the outcome.
|
||||
|
||||
|
||||
|
||||
Generate YAML File
|
||||
^^^^^^^^^^^^^^^^^^
|
||||
|
||||
.. code-block:: language
|
||||
|
||||
designate-manage pool generate_file [options]
|
||||
|
||||
Options:
|
||||
""""""""
|
||||
|
||||
--file YAML file output too (Default: ``/etc/designate/pools.yaml``)
|
||||
|
||||
Generate YAML File from Liberty Config
|
||||
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||
|
||||
.. code-block:: language
|
||||
|
||||
designate-manage pool export_from_config [options]
|
||||
|
||||
Options:
|
||||
""""""""
|
||||
|
||||
--file YAML file output too (Default: ``/etc/designate/pools.yaml``)
|
||||
|
94
doc/source/upgrade/mitaka.rst
Normal file
94
doc/source/upgrade/mitaka.rst
Normal file
@ -0,0 +1,94 @@
|
||||
..
|
||||
Copyright 2016 Hewlett Packard Enterprise Development Company, L.P.
|
||||
|
||||
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.
|
||||
|
||||
********************************
|
||||
Upgrading to Mitaka from Liberty
|
||||
********************************
|
||||
|
||||
.. note:: This is a WIP - it will be updated as more items are added to Mitaka
|
||||
|
||||
Pools Configuration
|
||||
===================
|
||||
|
||||
We have updated how the config data for pools is now stored.
|
||||
|
||||
Previously there was a mix of content in the ``designate.conf`` file and in the
|
||||
designate database.
|
||||
|
||||
We have moved all of the data to the database in Mitaka, to avoid confusion, and
|
||||
avoid the massive complexity that exists in the config file.
|
||||
|
||||
.. warning:: This part of the upgrade **requires** downtime.
|
||||
|
||||
We have 2 new commands in the ``designate-manage`` utility that are able to assist
|
||||
the migration.
|
||||
|
||||
To make the config syntax simpler we have a new YAML based config file that is
|
||||
used to load information into the database.
|
||||
|
||||
.. literalinclude:: ../../../etc/designate/pools.yaml.sample
|
||||
:language: yaml
|
||||
|
||||
We have a command that will allow you to take your current running config, and
|
||||
export it to the new YAML format.
|
||||
|
||||
.. note::
|
||||
|
||||
You will need to have at least one instance of central running, and machine
|
||||
``designate-manage`` is running on will need access to the messageing queue
|
||||
|
||||
.. code-block:: console
|
||||
|
||||
designate-manage pool generate_file --file output.yml
|
||||
|
||||
This will create a YAML file, with all the currently defined pools, and all of their config.
|
||||
|
||||
We suggest this is then migrated into a config management system, or other document management system.
|
||||
|
||||
From this point on all updated to pools should be one by updating this file, and
|
||||
running:
|
||||
|
||||
.. code-block:: console
|
||||
|
||||
designate-manage pool update --file /path/to/file.yml
|
||||
|
||||
|
||||
Pools - Step by Step
|
||||
--------------------
|
||||
|
||||
1. Ensure there is not 2 pools with the same name.
|
||||
2. Stop all Designate Services.
|
||||
3. Deploy new Mitaka code
|
||||
4. Start ``designate-central``
|
||||
5. Run
|
||||
.. code-block:: console
|
||||
|
||||
designate-manage pool export_from_config --file output.yml
|
||||
|
||||
6. Ensure the output file is correct (reference sample file for each value)
|
||||
7. Run
|
||||
|
||||
.. code-block:: console
|
||||
|
||||
designate-manage pool update --file output.yml --dry_run True --delete True
|
||||
|
||||
8. Ensure the output of this command is not removing any Pools
|
||||
9. Run
|
||||
|
||||
.. code-block:: console
|
||||
|
||||
designate-manage pool generate_file --file output.yml --delete True
|
||||
|
||||
10. Start the remaining designate services.
|
54
etc/designate/pools.yaml.sample
Normal file
54
etc/designate/pools.yaml.sample
Normal file
@ -0,0 +1,54 @@
|
||||
---
|
||||
|
||||
- name: default
|
||||
# The name is immutable. There will be no option to change the name after
|
||||
# creation and the only way will to change it will be to delete it
|
||||
# (and all zones associated with it) and recreate it.
|
||||
description: Default PowerDNS Pool
|
||||
|
||||
|
||||
# Attributes are Key:Value pairs that describe the pool. for example the level
|
||||
# of service (i.e. service_tier:GOLD), capabilities (i.e. anycast: true) or
|
||||
# other metadata. Users can use this infomation to point their zones to the
|
||||
# correct pool
|
||||
attributes: {}
|
||||
|
||||
# List out the NS records for zones hosted within this pool
|
||||
ns_records:
|
||||
- hostname: ns1-1.example.org.
|
||||
priority: 1
|
||||
- hostname: ns1-2.example.org.
|
||||
priority: 2
|
||||
|
||||
# List out the nameservers for this pool. These are the actual PowerDNS
|
||||
# servers. We use these to verify changes have propagated to all nameservers.
|
||||
nameservers:
|
||||
- host: 192.0.2.2
|
||||
port: 53
|
||||
- host: 192.0.2.3
|
||||
port: 53
|
||||
|
||||
# List out the targets for this pool. For PowerDNS, this is the database
|
||||
# (or databases, if you deploy a separate DB for each PowerDNS server)
|
||||
targets:
|
||||
- type: powerdns
|
||||
description: PowerDNS Database Cluster
|
||||
|
||||
# List out the designate-mdns servers from which PowerDNS servers should
|
||||
# request zone transfers (AXFRs) from.
|
||||
masters:
|
||||
- host: 192.0.2.1
|
||||
port: 5354
|
||||
|
||||
# PowerDNS Configuration options
|
||||
options:
|
||||
host: 192.0.2.1
|
||||
port: 53
|
||||
connection: 'mysql+pymysql://designate:password@127.0.0.1/designate_pdns?charset=utf8'
|
||||
|
||||
# Optional list of additional IP/Port's for which designate-mdns will send
|
||||
# DNS NOTIFY packets to
|
||||
also_notifies:
|
||||
- host: 192.0.2.4
|
||||
port: 53
|
||||
|
66
etc/designate/pools.yaml.sample-bind
Normal file
66
etc/designate/pools.yaml.sample-bind
Normal file
@ -0,0 +1,66 @@
|
||||
|
||||
- name: default-bind
|
||||
# The name is immutable. There will be no option to change the name after
|
||||
# creation and the only way will to change it will be to delete it
|
||||
# (and all zones associated with it) and recreate it.
|
||||
description: Default BIND9 Pool
|
||||
|
||||
attributes: {}
|
||||
|
||||
# List out the NS records for zones hosted within this pool
|
||||
ns_records:
|
||||
- hostname: ns1-1.example.org.
|
||||
priority: 1
|
||||
- hostname: ns1-2.example.org.
|
||||
priority: 2
|
||||
|
||||
# List out the nameservers for this pool. These are the actual BIND servers.
|
||||
# We use these to verify changes have propagated to all nameservers.
|
||||
nameservers:
|
||||
- host: 192.0.2.2
|
||||
port: 53
|
||||
- host: 192.0.2.3
|
||||
port: 53
|
||||
|
||||
# List out the targets for this pool. For BIND, most often, there will be one
|
||||
# entry for each BIND server.
|
||||
targets:
|
||||
- type: bind
|
||||
description: BIND9 Server 1
|
||||
|
||||
# List out the designate-mdns servers from which BIND servers should
|
||||
# request zone transfers (AXFRs) from.
|
||||
masters:
|
||||
- host: 192.0.2.1
|
||||
port: 5354
|
||||
|
||||
# BIND Configuration options
|
||||
options:
|
||||
host: 192.0.2.2
|
||||
port: 53
|
||||
rndc_host: 192.0.2.2
|
||||
rndc_port: 953
|
||||
rndc_key_file: /etc/designate/rndc.key
|
||||
|
||||
- type: bind
|
||||
description: BIND9 Server 2
|
||||
|
||||
# List out the designate-mdns servers from which BIND servers should
|
||||
# request zone transfers (AXFRs) from.
|
||||
masters:
|
||||
- host: 192.0.2.1
|
||||
port: 5354
|
||||
|
||||
# BIND Configuration options
|
||||
options:
|
||||
host: 192.0.2.3
|
||||
port: 53
|
||||
rndc_host: 192.0.2.3
|
||||
rndc_port: 953
|
||||
rndc_key_file: /etc/designate/rndc.key
|
||||
|
||||
# Optional list of additional IP/Port's for which designate-mdns will send
|
||||
# DNS NOTIFY packets to
|
||||
# also_notifies:
|
||||
# - host: 192.0.2.4
|
||||
# port: 53
|
115
etc/designate/pools.yaml.sample-multiple-pools
Normal file
115
etc/designate/pools.yaml.sample-multiple-pools
Normal file
@ -0,0 +1,115 @@
|
||||
---
|
||||
- name: pool-1
|
||||
# The name is immutable. There will be no option to change the name after
|
||||
# creation and the only way will to change it will be to delete it
|
||||
# (and all zones associated with it) and recreate it.
|
||||
description: Default PowerDNS Pool
|
||||
|
||||
attributes:
|
||||
internal: true
|
||||
|
||||
# List out the NS records for zones hosted within this pool
|
||||
ns_records:
|
||||
- hostname: ns1-1.example.org.
|
||||
priority: 1
|
||||
- hostname: ns1-2.example.org.
|
||||
priority: 2
|
||||
|
||||
# List out the nameservers for this pool. These are the actual PowerDNS
|
||||
# servers. We use these to verify changes have propagated to all nameservers.
|
||||
nameservers:
|
||||
- host: 192.0.2.2
|
||||
port: 53
|
||||
- host: 192.0.2.3
|
||||
port: 53
|
||||
|
||||
# List out the targets for this pool. For PowerDNS, this is the database
|
||||
# (or databases, if you deploy a separate DB for each PowerDNS server)
|
||||
targets:
|
||||
- type: powerdns
|
||||
description: PowerDNS Database Cluster
|
||||
|
||||
# List out the designate-mdns servers from which PowerDNS servers should
|
||||
# request zone transfers (AXFRs) from.
|
||||
masters:
|
||||
- host: 192.0.2.1
|
||||
port: 5354
|
||||
|
||||
# PowerDNS Configuration options
|
||||
options:
|
||||
host: 192.0.2.2
|
||||
port: 53
|
||||
connection: 'mysql+pymysql://designate:password@127.0.0.1/designate_pdns?charset=utf8'
|
||||
|
||||
# Optional list of additional IP/Port's for which designate-mdns will send
|
||||
# DNS NOTIFY packets to
|
||||
also_notifies:
|
||||
- host: 192.0.2.4
|
||||
port: 53
|
||||
|
||||
- name: pool-2
|
||||
# The name is immutable. There will be no option to change the name after
|
||||
# creation and the only way will to change it will be to delete it
|
||||
# (and all zones associated with it) and recreate it.
|
||||
description: Default PowerDNS Pool
|
||||
|
||||
attributes:
|
||||
external: true
|
||||
|
||||
# List out the NS records for zones hosted within this pool
|
||||
ns_records:
|
||||
- hostname: ns1-1.example.org.
|
||||
priority: 1
|
||||
- hostname: ns1-2.example.org.
|
||||
priority: 2
|
||||
|
||||
# List out the nameservers for this pool. These are the actual BIND servers.
|
||||
# We use these to verify changes have propagated to all nameservers.
|
||||
nameservers:
|
||||
- host: 192.0.2.2
|
||||
port: 53
|
||||
- host: 192.0.2.3
|
||||
port: 53
|
||||
|
||||
# List out the targets for this pool. For BIND, most often, there will be one
|
||||
# entry for each BIND server.
|
||||
targets:
|
||||
- type: bind
|
||||
description: BIND9 Server 1
|
||||
|
||||
# List out the designate-mdns servers from which BIND servers should
|
||||
# request zone transfers (AXFRs) from.
|
||||
masters:
|
||||
- host: 192.0.2.1
|
||||
port: 5354
|
||||
|
||||
# BIND Configuration options
|
||||
options:
|
||||
host: 192.0.2.2
|
||||
port: 53
|
||||
rndc_host: 192.0.2.2
|
||||
rndc_port: 953
|
||||
rndc_key_file: /etc/designate/rndc.key
|
||||
|
||||
- type: bind
|
||||
description: BIND9 Server 2
|
||||
|
||||
# List out the designate-mdns servers from which BIND servers should
|
||||
# request zone transfers (AXFRs) from.
|
||||
masters:
|
||||
- host: 192.0.2.1
|
||||
port: 5354
|
||||
|
||||
# BIND Configuration options
|
||||
options:
|
||||
host: 192.0.2.3
|
||||
port: 53
|
||||
rndc_host: 192.0.2.3
|
||||
rndc_port: 953
|
||||
rndc_key_file: /etc/designate/rndc.key
|
||||
|
||||
# Optional list of additional IP/Port's for which designate-mdns will send
|
||||
# DNS NOTIFY packets to
|
||||
also_notifies:
|
||||
- host: 192.0.2.4
|
||||
port:
|
14
releasenotes/notes/pool-config-db-45a2cad74e22d95e.yaml
Normal file
14
releasenotes/notes/pool-config-db-45a2cad74e22d95e.yaml
Normal file
@ -0,0 +1,14 @@
|
||||
---
|
||||
features:
|
||||
- New method of updating Pools.
|
||||
We have a new sub command on the ``designate-manage`` utility.
|
||||
upgrade:
|
||||
- Upgrade from Liberty to Mitaka *will* require downtime due to migration
|
||||
of Pool Configuration data from config files, to the database.
|
||||
See the Upgrade Documentation for full details.
|
||||
- Pool Names are now required to be unique. If there is 2 pools with the same
|
||||
name - this will need to be changed before the upgrade is performed.
|
||||
deprecations:
|
||||
- Create / Update / Delete API methods on /v2/pools/
|
||||
Using these messages **will** cause unforseen issues with pools. We highly recommend
|
||||
that the policy file be updated to restrict create / update / delete from all users
|
Loading…
x
Reference in New Issue
Block a user