python-troveclient/troveclient/osc/v1/database_backups.py
Lingxian Kong cd68986948 Get backup list by instance ID or name
Change-Id: I97d3fcfa2935f76cb8753734fec6dab8669e732a
2021-05-19 20:43:06 +12:00

341 lines
12 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 Backups action implementations"""
from osc_lib.command import command
from osc_lib import exceptions
from osc_lib import utils as osc_utils
from oslo_utils import uuidutils
from troveclient.i18n import _
from troveclient.osc.v1 import base
from troveclient import utils as trove_utils
def set_attributes_for_print_detail(backup):
info = backup._info.copy()
if hasattr(backup, 'datastore'):
info['datastore'] = backup.datastore['type']
info['datastore_version'] = backup.datastore['version']
info['datastore_version_id'] = backup.datastore['version_id']
return info
class ListDatabaseBackups(command.Lister):
_description = _("List database backups")
columns = ['ID', 'Instance ID', 'Name', 'Status', 'Parent ID',
'Updated', 'Project ID']
def get_parser(self, prog_name):
parser = super(ListDatabaseBackups, self).get_parser(prog_name)
parser.add_argument(
'--limit',
dest='limit',
metavar='<limit>',
default=None,
help=_('Return up to N number of the most recent bcakups.')
)
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.')
)
parser.add_argument(
'--datastore',
dest='datastore',
metavar='<datastore>',
default=None,
help=_('ID or name of the datastore (to filter backups by).')
)
parser.add_argument(
'--instance-id',
default=None,
help=_('Filter backups by database instance ID. Deprecated since '
'Xena. Use -i/--instance instead.')
)
parser.add_argument(
'-i',
'--instance',
default=None,
help=_('Filter backups by database instance(ID or name).')
)
parser.add_argument(
'--all-projects',
action='store_true',
help=_('Get all the backups of all the projects(Admin only).')
)
parser.add_argument(
'--project-id',
default=None,
help=_('Filter backups by project ID.')
)
return parser
def take_action(self, parsed_args):
database_backups = self.app.client_manager.database.backups
instance_id = parsed_args.instance or parsed_args.instance_id
if instance_id:
instance_mgr = self.app.client_manager.database.instances
instance_id = trove_utils.get_resource_id(instance_mgr,
instance_id)
items = database_backups.list(limit=parsed_args.limit,
datastore=parsed_args.datastore,
marker=parsed_args.marker,
instance_id=instance_id,
all_projects=parsed_args.all_projects,
project_id=parsed_args.project_id)
backups = items
while items.next and not parsed_args.limit:
items = database_backups.list(
marker=items.next,
datastore=parsed_args.datastore,
instance_id=parsed_args.instance_id,
all_projects=parsed_args.all_projects,
project_id=parsed_args.project_id
)
backups += items
backups = [osc_utils.get_item_properties(b, self.columns)
for b in backups]
return self.columns, backups
class ListDatabaseInstanceBackups(command.Lister):
_description = _("Lists available backups for an instance.")
columns = ['ID', 'Instance ID', 'Name', 'Status', 'Parent ID',
'Updated']
def get_parser(self, prog_name):
parser = super(ListDatabaseInstanceBackups, self).get_parser(prog_name)
parser.add_argument(
'instance',
metavar='<instance>',
help=_('ID or name of the instance.')
)
parser.add_argument(
'--limit',
dest='limit',
metavar='<limit>',
default=None,
help=_('Return up to N number of the most recent bcakups.')
)
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_instances = self.app.client_manager.database.instances
instance = osc_utils.find_resource(database_instances,
parsed_args.instance)
items = database_instances.backups(instance, limit=parsed_args.limit,
marker=parsed_args.marker)
backups = items
while items.next and not parsed_args.limit:
items = database_instances.backups(instance, marker=items.next)
backups += items
backups = [osc_utils.get_item_properties(b, self.columns)
for b in backups]
return self.columns, backups
class ShowDatabaseBackup(command.ShowOne):
_description = _("Shows details of a database backup")
def get_parser(self, prog_name):
parser = super(ShowDatabaseBackup, self).get_parser(prog_name)
parser.add_argument(
'backup',
metavar='<backup>',
help=_('ID or name of the backup'),
)
return parser
def take_action(self, parsed_args):
database_backups = self.app.client_manager.database.backups
backup = osc_utils.find_resource(database_backups, parsed_args.backup)
backup = set_attributes_for_print_detail(backup)
return zip(*sorted(backup.items()))
class DeleteDatabaseBackup(base.TroveDeleter):
_description = _("Deletes a backup.")
def get_parser(self, prog_name):
parser = super(DeleteDatabaseBackup, self).get_parser(prog_name)
parser.add_argument(
'backup',
nargs='+',
metavar='backup',
help='Id or name of backup(s).'
)
return parser
def take_action(self, parsed_args):
db_backups = self.app.client_manager.database.backups
# Used for batch deletion
self.delete_func = db_backups.delete
self.resource = 'database backup'
ids = []
for backup_id in parsed_args.backup:
if not uuidutils.is_uuid_like(backup_id):
try:
backup_id = trove_utils.get_resource_id_by_name(
db_backups, backup_id
)
except Exception as e:
msg = ("Failed to get database backup %s, error: %s" %
(backup_id, str(e)))
raise exceptions.CommandError(msg)
ids.append(backup_id)
self.delete_resources(ids)
class CreateDatabaseBackup(command.ShowOne):
_description = _("Creates a backup of an instance.")
def get_parser(self, prog_name):
parser = super(CreateDatabaseBackup, self).get_parser(prog_name)
parser.add_argument(
'name',
metavar='<name>',
help=_('Name of the backup.')
)
parser.add_argument(
'-i',
'--instance',
metavar='<instance>',
help=_('ID or name of the instance. This is not required if '
'restoring a backup from the data location.')
)
parser.add_argument(
'--description',
metavar='<description>',
default=None,
help=_('An optional description for the backup.')
)
parser.add_argument(
'--parent',
metavar='<parent>',
default=None,
help=_('Optional ID of the parent backup to perform an'
' incremental backup from.')
)
parser.add_argument(
'--incremental',
action='store_true',
default=False,
help=_('Create an incremental backup based on the last'
' full or incremental backup. It will create a'
' full backup if no existing backup found.')
)
parser.add_argument(
'--swift-container',
help=_('The container name for storing the backup data when Swift '
'is used as backup storage backend. If not specified, will '
'use the container name configured in the backup strategy, '
'otherwise, the default value configured by the cloud '
'operator. Non-existent container is created '
'automatically.')
)
parser.add_argument(
'--restore-from',
help=_('The original backup data location, typically this is a '
'Swift object URL.')
)
parser.add_argument(
'--restore-datastore-version',
help=_('ID of the local datastore version corresponding to the '
'original backup')
)
parser.add_argument(
'--restore-size', type=float,
help=_('The original backup size.')
)
return parser
def take_action(self, parsed_args):
manager = self.app.client_manager.database
database_backups = manager.backups
params = {}
instance_id = None
if parsed_args.restore_from:
# Leave the input validation to Trove server.
params.update({
'restore_from': parsed_args.restore_from,
'restore_ds_version': parsed_args.restore_datastore_version,
'restore_size': parsed_args.restore_size,
})
elif not parsed_args.instance:
raise exceptions.CommandError('Instance ID or name is required if '
'not restoring a backup.')
else:
instance_id = trove_utils.get_resource_id(manager.instances,
parsed_args.instance)
params.update({
'description': parsed_args.description,
'parent_id': parsed_args.parent,
'incremental': parsed_args.incremental,
'swift_container': parsed_args.swift_container
})
backup = database_backups.create(parsed_args.name, instance_id,
**params)
backup = set_attributes_for_print_detail(backup)
return zip(*sorted(backup.items()))
class DeleteDatabaseBackupExecution(command.Command):
_description = _("Deletes an execution.")
def get_parser(self, prog_name):
parser = super(DeleteDatabaseBackupExecution, self).get_parser(
prog_name)
parser.add_argument(
'execution',
metavar='<execution>',
help=_('ID of the execution to delete.')
)
return parser
def take_action(self, parsed_args):
database_backups = self.app.client_manager.database.backups
database_backups.execution_delete(parsed_args.execution)