"""
Copyright 2015 Rackspace
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 logging
import os
from tempest.lib.cli import base
from designateclient.functionaltests.config import cfg
from designateclient.functionaltests.models import FieldValueModel
from designateclient.functionaltests.models import ListModel
LOG = logging.getLogger(__name__)
[docs]def build_option_string(options):
"""Format a string of option flags (--key 'value').
This will quote the values, in case spaces are included.
Any values that are None are excluded entirely.
Usage:
build_option_string({
"--email": "me@example.com",
"--name": "example.com."
"--ttl": None,
})
Returns:
"--email 'me@example.com' --name 'example.com.'
"""
return " ".join("{0} '{1}'".format(flag, value)
for flag, value in options.items()
if value is not None)
[docs]def build_flags_string(flags):
"""Format a string of value-less flags.
Pass in a dictionary mapping flags to booleans. Those flags set to true
are included in the returned string.
Usage:
build_flags_string({
'--no-ttl': True,
'--no-name': False,
'--verbose': True,
})
Returns:
'--no-ttl --verbose'
"""
flags = {flag: is_set for flag, is_set in flags.items() if is_set}
return " ".join(flags.keys())
[docs]class ZoneCommands(object):
"""This is a mixin that provides zone commands to DesignateCLI"""
[docs] def zone_list(self, *args, **kwargs):
return self.parsed_cmd('zone list', ListModel, *args, **kwargs)
[docs] def zone_show(self, id, *args, **kwargs):
return self.parsed_cmd('zone show %s' % id, FieldValueModel, *args,
**kwargs)
[docs] def zone_delete(self, id, *args, **kwargs):
return self.parsed_cmd('zone delete %s' % id, FieldValueModel, *args,
**kwargs)
[docs] def zone_create(self, name, email=None, ttl=None, description=None,
type=None, masters=None, *args, **kwargs):
options_str = build_option_string({
"--email": email,
"--ttl": ttl,
"--description": description,
"--masters": masters,
"--type": type,
})
cmd = 'zone create {0} {1}'.format(name, options_str)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs] def zone_set(self, id, email=None, ttl=None, description=None,
type=None, masters=None, *args, **kwargs):
options_str = build_option_string({
"--email": email,
"--ttl": ttl,
"--description": description,
"--masters": masters,
"--type": type,
})
cmd = 'zone set {0} {1}'.format(id, options_str)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs]class ZoneTransferCommands(object):
"""A mixin for DesignateCLI to add zone transfer commands"""
[docs] def zone_transfer_request_list(self, *args, **kwargs):
cmd = 'zone transfer request list'
return self.parsed_cmd(cmd, ListModel, *args, **kwargs)
[docs] def zone_transfer_request_create(self, zone_id, target_project_id=None,
description=None, *args, **kwargs):
options_str = build_option_string({
"--target-project-id": target_project_id,
"--description": description,
})
cmd = 'zone transfer request create {0} {1}'.format(
zone_id, options_str)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs] def zone_transfer_request_show(self, id, *args, **kwargs):
cmd = 'zone transfer request show {0}'.format(id)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs] def zone_transfer_request_set(self, id, description=None, *args, **kwargs):
options_str = build_option_string({"--description": description})
cmd = 'zone transfer request set {0} {1}'.format(options_str, id)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs] def zone_transfer_request_delete(self, id, *args, **kwargs):
cmd = 'zone transfer request delete {0}'.format(id)
return self.parsed_cmd(cmd, *args, **kwargs)
[docs] def zone_transfer_accept_request(self, id, key, *args, **kwargs):
options_str = build_option_string({
"--transfer-id": id,
"--key": key,
})
cmd = 'zone transfer accept request {0}'.format(options_str)
return self.parsed_cmd(cmd, *args, **kwargs)
[docs] def zone_transfer_accept_show(self, id, *args, **kwargs):
cmd = 'zone transfer accept show {0}'.format(id)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs]class ZoneExportCommands(object):
"""A mixin for DesignateCLI to add zone export commands"""
[docs] def zone_export_list(self, *args, **kwargs):
cmd = 'zone export list'
return self.parsed_cmd(cmd, ListModel, *args, **kwargs)
[docs] def zone_export_create(self, zone_id, *args, **kwargs):
cmd = 'zone export create {0}'.format(
zone_id)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs] def zone_export_show(self, zone_export_id, *args, **kwargs):
cmd = 'zone export show {0}'.format(zone_export_id)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs] def zone_export_delete(self, zone_export_id, *args, **kwargs):
cmd = 'zone export delete {0}'.format(zone_export_id)
return self.parsed_cmd(cmd, *args, **kwargs)
[docs] def zone_export_showfile(self, zone_export_id, *args, **kwargs):
cmd = 'zone export showfile {0}'.format(zone_export_id)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs]class ZoneImportCommands(object):
"""A mixin for DesignateCLI to add zone import commands"""
[docs] def zone_import_list(self, *args, **kwargs):
cmd = 'zone import list'
return self.parsed_cmd(cmd, ListModel, *args, **kwargs)
[docs] def zone_import_create(self, zone_file_path, *args, **kwargs):
cmd = 'zone import create {0}'.format(zone_file_path)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs] def zone_import_show(self, zone_import_id, *args, **kwargs):
cmd = 'zone import show {0}'.format(zone_import_id)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs] def zone_import_delete(self, zone_import_id, *args, **kwargs):
cmd = 'zone import delete {0}'.format(zone_import_id)
return self.parsed_cmd(cmd, *args, **kwargs)
[docs]class RecordsetCommands(object):
[docs] def recordset_show(self, zone_id, id, *args, **kwargs):
cmd = 'recordset show {0} {1}'.format(zone_id, id)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs] def recordset_list(self, zone_id, *args, **kwargs):
cmd = 'recordset list {0}'.format(zone_id)
return self.parsed_cmd(cmd, ListModel, *args, **kwargs)
[docs] def recordset_create(self, zone_id, name, records=None, type=None,
description=None, ttl=None, *args, **kwargs):
options_str = build_option_string({
'--records': records,
'--type': type,
'--description': description,
'--ttl': ttl,
})
cmd = 'recordset create {0} {1} {2}'.format(zone_id, name, options_str)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs] def recordset_set(self, zone_id, id, records=None, type=None,
description=None, ttl=None, no_description=False,
no_ttl=False, *args, **kwargs):
options_str = build_option_string({
'--records': records,
'--type': type,
'--description': description,
'--ttl': ttl,
})
flags_str = build_flags_string({
'--no-description': no_description,
'--no-ttl': no_ttl,
})
cmd = 'recordset set {0} {1} {2} {3}'.format(
zone_id, id, flags_str, options_str)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs] def recordset_delete(self, zone_id, id, *args, **kwargs):
cmd = 'recordset delete {0} {1}'.format(zone_id, id)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs]class TLDCommands(object):
[docs] def tld_list(self, *args, **kwargs):
return self.parsed_cmd('tld list', ListModel, *args, **kwargs)
[docs] def tld_show(self, id, *args, **kwargs):
return self.parsed_cmd('tld show {0}'.format(id), FieldValueModel,
*args, **kwargs)
[docs] def tld_delete(self, id, *args, **kwargs):
return self.parsed_cmd('tld delete {0}'.format(id), *args, **kwargs)
[docs] def tld_create(self, name, description=None, *args, **kwargs):
options_str = build_option_string({
'--name': name,
'--description': description,
})
cmd = 'tld create {0}'.format(options_str)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs] def tld_set(self, id, name=None, description=None, no_description=False,
*args, **kwargs):
options_str = build_option_string({
'--name': name,
'--description': description,
})
flags_str = build_flags_string({'--no-description': no_description})
cmd = 'tld set {0} {1} {2}'.format(id, options_str, flags_str)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs]class BlacklistCommands(object):
[docs] def zone_blacklist_list(self, *args, **kwargs):
cmd = 'zone blacklist list'
return self.parsed_cmd(cmd, ListModel, *args, **kwargs)
[docs] def zone_blacklist_create(self, pattern, description=None, *args,
**kwargs):
options_str = build_option_string({
'--pattern': pattern,
'--description': description,
})
cmd = 'zone blacklist create {0}'.format(options_str)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs] def zone_blacklist_set(self, id, pattern=None, description=None,
no_description=False, *args, **kwargs):
options_str = build_option_string({
'--pattern': pattern,
'--description': description,
})
flags_str = build_flags_string({'--no-description': no_description})
cmd = 'zone blacklist set {0} {1} {2}'.format(id, options_str,
flags_str)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs] def zone_blacklist_show(self, id, *args, **kwargs):
cmd = 'zone blacklist show {0}'.format(id)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs] def zone_blacklist_delete(self, id, *args, **kwargs):
cmd = 'zone blacklist delete {0}'.format(id)
return self.parsed_cmd(cmd, FieldValueModel, *args, **kwargs)
[docs]class DesignateCLI(base.CLIClient, ZoneCommands, ZoneTransferCommands,
ZoneExportCommands, ZoneImportCommands, RecordsetCommands,
TLDCommands, BlacklistCommands):
# instantiate this once to minimize requests to keystone
_CLIENTS = None
def __init__(self, *args, **kwargs):
super(DesignateCLI, self).__init__(*args, **kwargs)
# grab the project id. this is used for zone transfer requests
resp = FieldValueModel(self.openstack('token issue'))
self.project_id = resp.project_id
@property
def using_auth_override(self):
return bool(cfg.CONF.identity.override_endpoint)
@classmethod
[docs] def get_clients(cls):
if not cls._CLIENTS:
cls._init_clients()
return cls._CLIENTS
@classmethod
def _init_clients(cls):
cls._CLIENTS = {
'default': DesignateCLI(
cli_dir=cfg.CONF.designateclient.directory,
username=cfg.CONF.identity.username,
password=cfg.CONF.identity.password,
tenant_name=cfg.CONF.identity.tenant_name,
uri=cfg.CONF.identity.uri,
),
'alt': DesignateCLI(
cli_dir=cfg.CONF.designateclient.directory,
username=cfg.CONF.identity.alt_username,
password=cfg.CONF.identity.alt_password,
tenant_name=cfg.CONF.identity.alt_tenant_name,
uri=cfg.CONF.identity.uri,
),
'admin': DesignateCLI(
cli_dir=cfg.CONF.designateclient.directory,
username=cfg.CONF.identity.admin_username,
password=cfg.CONF.identity.admin_password,
tenant_name=cfg.CONF.identity.admin_tenant_name,
uri=cfg.CONF.identity.uri,
)
}
@classmethod
[docs] def as_user(self, user):
clients = self.get_clients()
if user in clients:
return clients[user]
raise Exception("User '{0}' does not exist".format(user))
[docs] def parsed_cmd(self, cmd, model=None, *args, **kwargs):
if self.using_auth_override:
# use --os-url and --os-token
func = self._openstack_noauth
else:
# use --os-username --os-tenant-name --os-password --os-auth-url
func = self.openstack
out = func(cmd, *args, **kwargs)
LOG.debug(out)
if model is not None:
return model(out)
return out
def _openstack_noauth(self, cmd, *args, **kwargs):
exe = os.path.join(cfg.CONF.designateclient.directory, 'openstack')
options = build_option_string({
'--os-url': cfg.CONF.identity.override_endpoint,
'--os-token': cfg.CONF.identity.override_token,
})
cmd = options + " " + cmd
return base.execute(exe, cmd, *args, **kwargs)