
This PR makes troveclient to handle API response in new data schema.
Trove API response data schema has changed by the commit[1].
Original problem is that python-troveclient can't parse a API response
data in new data schema, resulting in an error.
[1]: 429c39890e
Task: 44986
Story: 2009979
Change-Id: I2e446c68c3b82c11d13f6bace54273c109e02069
413 lines
14 KiB
Python
413 lines
14 KiB
Python
# 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.
|
|
|
|
"""Database v1 Clusters action implementations"""
|
|
|
|
from osc_lib.command import command
|
|
from osc_lib import exceptions
|
|
from osc_lib import utils
|
|
|
|
from troveclient.i18n import _
|
|
from troveclient.v1.shell import _parse_extended_properties
|
|
from troveclient.v1.shell import _parse_instance_options
|
|
from troveclient.v1.shell import EXT_PROPS_HELP
|
|
from troveclient.v1.shell import EXT_PROPS_METAVAR
|
|
from troveclient.v1.shell import INSTANCE_HELP
|
|
from troveclient.v1.shell import INSTANCE_METAVAR
|
|
|
|
|
|
def set_attributes_for_print_detail(cluster):
|
|
info = cluster._info.copy()
|
|
if hasattr(cluster, 'datastore'):
|
|
info['datastore'] = cluster.datastore['type']
|
|
info['datastore_version'] = cluster.datastore['version']
|
|
if hasattr(cluster, 'task'):
|
|
info['task_description'] = cluster.task['description']
|
|
info['task_name'] = cluster.task['name']
|
|
info.pop('task', None)
|
|
if hasattr(cluster, 'ip'):
|
|
ip = []
|
|
for addr in cluster.ip:
|
|
if isinstance(addr, dict):
|
|
ip.append(addr['address'])
|
|
else:
|
|
ip.append(addr)
|
|
info['ip'] = ', '.join(ip)
|
|
instances = info.pop('instances', None)
|
|
if instances:
|
|
info['instance_count'] = len(instances)
|
|
info.pop('links', None)
|
|
return info
|
|
|
|
|
|
class ListDatabaseClusters(command.Lister):
|
|
|
|
_description = _("List database clusters")
|
|
columns = ['ID', 'Name', 'Datastore', 'Datastore Version',
|
|
'Task Name']
|
|
|
|
def get_parser(self, prog_name):
|
|
parser = super(ListDatabaseClusters, self).get_parser(prog_name)
|
|
parser.add_argument(
|
|
'--limit',
|
|
dest='limit',
|
|
metavar='<limit>',
|
|
type=int,
|
|
default=None,
|
|
help=_('Limit the number of results displayed.')
|
|
)
|
|
parser.add_argument(
|
|
'--marker',
|
|
dest='marker',
|
|
metavar='<ID>',
|
|
type=str,
|
|
default=None,
|
|
help=_('Begin displaying the results for IDs greater than the'
|
|
' specified marker. When used with ``--limit``, set '
|
|
' this to the last ID displayed in the previous run.')
|
|
)
|
|
return parser
|
|
|
|
def take_action(self, parsed_args):
|
|
database_clusters = self.app.client_manager.database.clusters
|
|
clusters = database_clusters.list(limit=parsed_args.limit,
|
|
marker=parsed_args.marker)
|
|
for cluster in clusters:
|
|
setattr(cluster, 'datastore_version',
|
|
cluster.datastore['version'])
|
|
setattr(cluster, 'datastore', cluster.datastore['type'])
|
|
setattr(cluster, 'task_name', cluster.task['name'])
|
|
|
|
clusters = [utils.get_item_properties(c, self.columns)
|
|
for c in clusters]
|
|
return self.columns, clusters
|
|
|
|
|
|
class ShowDatabaseCluster(command.ShowOne):
|
|
_description = _("Shows details of a database cluster")
|
|
|
|
def get_parser(self, prog_name):
|
|
parser = super(ShowDatabaseCluster, self).get_parser(prog_name)
|
|
parser.add_argument(
|
|
'cluster',
|
|
metavar='<cluster>',
|
|
help=_('ID or name of the cluster'),
|
|
)
|
|
return parser
|
|
|
|
def take_action(self, parsed_args):
|
|
database_clusters = self.app.client_manager.database.clusters
|
|
cluster = utils.find_resource(database_clusters, parsed_args.cluster)
|
|
cluster = set_attributes_for_print_detail(cluster)
|
|
return zip(*sorted(cluster.items()))
|
|
|
|
|
|
class DeleteDatabaseCluster(command.Command):
|
|
|
|
_description = _("Deletes a cluster.")
|
|
|
|
def get_parser(self, prog_name):
|
|
parser = super(DeleteDatabaseCluster, self).get_parser(prog_name)
|
|
parser.add_argument(
|
|
'cluster',
|
|
metavar='<cluster>',
|
|
help=_('ID or name of the cluster.'),
|
|
)
|
|
return parser
|
|
|
|
def take_action(self, parsed_args):
|
|
database_clusters = self.app.client_manager.database.clusters
|
|
try:
|
|
cluster = utils.find_resource(database_clusters,
|
|
parsed_args.cluster)
|
|
database_clusters.delete(cluster)
|
|
except Exception as e:
|
|
msg = (_("Failed to delete cluster %(cluster)s: %(e)s")
|
|
% {'cluster': parsed_args.cluster, 'e': e})
|
|
raise exceptions.CommandError(msg)
|
|
|
|
|
|
class CreateDatabaseCluster(command.ShowOne):
|
|
_description = _("Creates a new database cluster.")
|
|
|
|
def get_parser(self, prog_name):
|
|
parser = super(CreateDatabaseCluster, self).get_parser(prog_name)
|
|
parser.add_argument(
|
|
'name',
|
|
metavar='<name>',
|
|
type=str,
|
|
help=_('Name of the cluster.'),
|
|
)
|
|
parser.add_argument(
|
|
'datastore',
|
|
metavar='<datastore>',
|
|
help=_('A datastore name or ID.'),
|
|
)
|
|
parser.add_argument(
|
|
'datastore_version',
|
|
metavar='<datastore_version>',
|
|
help=_('A datastore version name or ID.'),
|
|
)
|
|
parser.add_argument(
|
|
'--instance',
|
|
metavar=INSTANCE_METAVAR,
|
|
action='append',
|
|
dest='instances',
|
|
default=[],
|
|
help=INSTANCE_HELP,
|
|
)
|
|
parser.add_argument(
|
|
'--locality',
|
|
metavar='<policy>',
|
|
default=None,
|
|
choices=['affinity', 'anti-affinity'],
|
|
help=_('Locality policy to use when creating cluster. '
|
|
'Choose one of %(choices)s.'),
|
|
)
|
|
parser.add_argument(
|
|
'--extended-properties',
|
|
dest='extended_properties',
|
|
metavar=EXT_PROPS_METAVAR,
|
|
default=None,
|
|
help=EXT_PROPS_HELP,
|
|
)
|
|
parser.add_argument(
|
|
'--configuration',
|
|
metavar='<configuration>',
|
|
type=str,
|
|
default=None,
|
|
help=_('ID of the configuration group to attach to the cluster.'),
|
|
)
|
|
return parser
|
|
|
|
def take_action(self, parsed_args):
|
|
database = self.app.client_manager.database
|
|
instances = _parse_instance_options(database, parsed_args.instances)
|
|
extended_properties = {}
|
|
if parsed_args.extended_properties:
|
|
extended_properties = _parse_extended_properties(
|
|
parsed_args.extended_properties)
|
|
cluster = database.clusters.create(
|
|
parsed_args.name,
|
|
parsed_args.datastore,
|
|
parsed_args.datastore_version,
|
|
instances=instances,
|
|
locality=parsed_args.locality,
|
|
extended_properties=extended_properties,
|
|
configuration=parsed_args.configuration)
|
|
cluster = set_attributes_for_print_detail(cluster)
|
|
return zip(*sorted(cluster.items()))
|
|
|
|
|
|
class ResetDatabaseClusterStatus(command.Command):
|
|
|
|
_description = _("Set the cluster task to NONE.")
|
|
|
|
def get_parser(self, prog_name):
|
|
parser = super(ResetDatabaseClusterStatus, self).get_parser(prog_name)
|
|
parser.add_argument(
|
|
'cluster',
|
|
metavar='<cluster>',
|
|
help=_('ID or name of the cluster.'),
|
|
)
|
|
return parser
|
|
|
|
def take_action(self, parsed_args):
|
|
database_clusters = self.app.client_manager.database.clusters
|
|
cluster = utils.find_resource(database_clusters,
|
|
parsed_args.cluster)
|
|
database_clusters.reset_status(cluster)
|
|
|
|
|
|
class ListDatabaseClusterInstances(command.Lister):
|
|
|
|
_description = _("Lists all instances of a cluster.")
|
|
columns = ['ID', 'Name', 'Flavor ID', 'Size', 'Status']
|
|
|
|
def get_parser(self, prog_name):
|
|
parser = (super(ListDatabaseClusterInstances, self)
|
|
.get_parser(prog_name))
|
|
parser.add_argument(
|
|
'cluster',
|
|
metavar='<cluster>',
|
|
help=_('ID or name of the cluster.'))
|
|
return parser
|
|
|
|
def take_action(self, parsed_args):
|
|
database_clusters = self.app.client_manager.database.clusters
|
|
cluster = utils.find_resource(database_clusters, parsed_args.cluster)
|
|
instances = cluster._info['instances']
|
|
for instance in instances:
|
|
instance['flavor_id'] = instance['flavor']['id']
|
|
if instance.get('volume'):
|
|
instance['size'] = instance['volume']['size']
|
|
|
|
instances = [utils.get_dict_properties(inst, self.columns)
|
|
for inst in instances]
|
|
return self.columns, instances
|
|
|
|
|
|
class UpgradeDatabaseCluster(command.Command):
|
|
|
|
_description = _("Upgrades a cluster to a new datastore version.")
|
|
|
|
def get_parser(self, prog_name):
|
|
parser = super(UpgradeDatabaseCluster, self).get_parser(prog_name)
|
|
parser.add_argument(
|
|
'cluster',
|
|
metavar='<cluster>',
|
|
help=_('ID or name of the cluster.'),
|
|
)
|
|
parser.add_argument(
|
|
'datastore_version',
|
|
metavar='<datastore_version>',
|
|
help=_('A datastore version name or ID.'),
|
|
)
|
|
return parser
|
|
|
|
def take_action(self, parsed_args):
|
|
database_clusters = self.app.client_manager.database.clusters
|
|
cluster = utils.find_resource(database_clusters,
|
|
parsed_args.cluster)
|
|
database_clusters.upgrade(cluster, parsed_args.datastore_version)
|
|
|
|
|
|
class ForceDeleteDatabaseCluster(command.Command):
|
|
|
|
_description = _("Force delete a cluster.")
|
|
|
|
def get_parser(self, prog_name):
|
|
parser = super(ForceDeleteDatabaseCluster, self).get_parser(prog_name)
|
|
parser.add_argument(
|
|
'cluster',
|
|
metavar='<cluster>',
|
|
help=_('ID or name of the cluster.'),
|
|
)
|
|
return parser
|
|
|
|
def take_action(self, parsed_args):
|
|
database_clusters = self.app.client_manager.database.clusters
|
|
cluster = utils.find_resource(database_clusters,
|
|
parsed_args.cluster)
|
|
database_clusters.reset_status(cluster)
|
|
try:
|
|
database_clusters.delete(cluster)
|
|
except Exception as e:
|
|
msg = (_("Failed to delete cluster %(cluster)s: %(e)s")
|
|
% {'cluster': parsed_args.cluster, 'e': e})
|
|
raise exceptions.CommandError(msg)
|
|
|
|
|
|
class GrowDatabaseCluster(command.Command):
|
|
|
|
_description = _("Adds more instances to a cluster.")
|
|
|
|
def get_parser(self, prog_name):
|
|
parser = super(GrowDatabaseCluster, self).get_parser(prog_name)
|
|
parser.add_argument(
|
|
'--instance',
|
|
metavar=INSTANCE_METAVAR,
|
|
action='append',
|
|
dest='instances',
|
|
default=[],
|
|
help=INSTANCE_HELP
|
|
)
|
|
parser.add_argument(
|
|
'cluster',
|
|
metavar='<cluster>',
|
|
help=_('ID or name of the cluster.')
|
|
)
|
|
return parser
|
|
|
|
def take_action(self, parsed_args):
|
|
database_client_manager = self.app.client_manager.database
|
|
|
|
db_clusters = database_client_manager.clusters
|
|
cluster = utils.find_resource(db_clusters,
|
|
parsed_args.cluster)
|
|
|
|
instances = _parse_instance_options(database_client_manager,
|
|
parsed_args.instances,
|
|
for_grow=True)
|
|
db_clusters.grow(cluster, instances=instances)
|
|
|
|
|
|
class ShrinkDatabaseCluster(command.Command):
|
|
|
|
_description = _("Drops instances from a cluster.")
|
|
|
|
def get_parser(self, prog_name):
|
|
parser = super(ShrinkDatabaseCluster, self).get_parser(prog_name)
|
|
parser.add_argument(
|
|
'cluster',
|
|
metavar='<cluster>',
|
|
help=_('ID or name of the cluster.')
|
|
)
|
|
parser.add_argument(
|
|
'instances',
|
|
metavar='<instance>',
|
|
nargs='+',
|
|
default=[],
|
|
help=_("Drop instance(s) from the cluster. Specify "
|
|
"multiple ids to drop multiple instances.")
|
|
)
|
|
return parser
|
|
|
|
def take_action(self, parsed_args):
|
|
database_client_manager = self.app.client_manager.database
|
|
|
|
db_clusters = database_client_manager.clusters
|
|
cluster = utils.find_resource(db_clusters,
|
|
parsed_args.cluster)
|
|
|
|
db_instances = database_client_manager.instances
|
|
instances = [
|
|
{'id': utils.find_resource(db_instances,
|
|
instance).id}
|
|
for instance in parsed_args.instances
|
|
]
|
|
db_clusters.shrink(cluster, instances)
|
|
|
|
|
|
class ListDatabaseClusterModules(command.Lister):
|
|
|
|
_description = _("Lists all modules for each instance of a cluster.")
|
|
columns = ['instance_name', 'Module Name', 'Module Type', 'md5',
|
|
'created', 'updated']
|
|
|
|
def get_parser(self, prog_name):
|
|
parser = (super(ListDatabaseClusterModules, self)
|
|
.get_parser(prog_name))
|
|
parser.add_argument(
|
|
'cluster',
|
|
metavar='<cluster>',
|
|
help=_('ID or name of the cluster.'))
|
|
return parser
|
|
|
|
def take_action(self, parsed_args):
|
|
database_clusters = self.app.client_manager.database.clusters
|
|
database_instances = self.app.client_manager.database.instances
|
|
cluster = utils.find_resource(database_clusters, parsed_args.cluster)
|
|
instances = cluster._info['instances']
|
|
modules = []
|
|
for instance in instances:
|
|
new_list = database_instances.modules(instance['id'])
|
|
for item in new_list:
|
|
item.instance_id = instance['id']
|
|
item.instance_name = instance['name']
|
|
item.module_name = item.name
|
|
item.module_type = item.type
|
|
modules += new_list
|
|
modules = [utils.get_item_properties(module, self.columns)
|
|
for module in modules]
|
|
return self.columns, modules
|