Add OpenStackClient plugin for scheduledoperations
This patch adds data protection support to the python-openstackclient through a plugin for scheduledoperations. Co-Authored-By: Spencer Yu <yushb@gohighsec.com> Change-Id: Id2712e3a081e0fd2b69cad093d1a6b26b644e967 Partially-Implements: blueprint karbor-support-python-openstackclient
This commit is contained in:
208
karborclient/osc/v1/scheduled_operations.py
Normal file
208
karborclient/osc/v1/scheduled_operations.py
Normal file
@@ -0,0 +1,208 @@
|
||||
# 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.
|
||||
|
||||
"""Data protection V1 scheduled_operations action implementations"""
|
||||
|
||||
import six
|
||||
|
||||
from oslo_utils import uuidutils
|
||||
|
||||
from osc_lib.command import command
|
||||
from osc_lib import utils as osc_utils
|
||||
from oslo_log import log as logging
|
||||
|
||||
from karborclient.common.apiclient import exceptions
|
||||
from karborclient.i18n import _
|
||||
|
||||
|
||||
class ListScheduledOperations(command.Lister):
|
||||
_description = _("List scheduled_operations.")
|
||||
|
||||
log = logging.getLogger(__name__ + ".ListScheduledOperations")
|
||||
|
||||
def get_parser(self, prog_name):
|
||||
parser = super(ListScheduledOperations, self).get_parser(prog_name)
|
||||
parser.add_argument(
|
||||
'--all-projects',
|
||||
action='store_true',
|
||||
default=False,
|
||||
help=_('Shows details for all tenants. Admin only.'),
|
||||
)
|
||||
parser.add_argument(
|
||||
'--name',
|
||||
metavar='<name>',
|
||||
help=_('Filters results by a name. Default=None.'),
|
||||
)
|
||||
parser.add_argument(
|
||||
'--operation_type',
|
||||
metavar='<operation_type>',
|
||||
default=None,
|
||||
help=_('Filters results by a type. Default=None.'),
|
||||
)
|
||||
parser.add_argument(
|
||||
'--trigger_id',
|
||||
metavar='<trigger_id>',
|
||||
default=None,
|
||||
help=_('Filters results by a trigger id. Default=None.'),
|
||||
)
|
||||
parser.add_argument(
|
||||
'--operation_definition',
|
||||
metavar='<operation_definition>',
|
||||
default=None,
|
||||
help=_('Filters results by a operation definition. Default=None.'),
|
||||
)
|
||||
parser.add_argument(
|
||||
'--marker',
|
||||
metavar='<scheduled_operations>',
|
||||
help=_('The last scheduled_operations ID of the previous page'),
|
||||
)
|
||||
parser.add_argument(
|
||||
'--limit',
|
||||
type=int,
|
||||
metavar='<num-scheduled_operations>',
|
||||
help=_('Maximum number of scheduled_operations to display'),
|
||||
)
|
||||
parser.add_argument(
|
||||
'--sort',
|
||||
metavar="<key>[:<direction>]",
|
||||
default=None,
|
||||
help=_("Sort output by selected keys and directions(asc or desc) "
|
||||
"(default: name:asc), multiple keys and directions can be "
|
||||
"specified separated by comma"),
|
||||
)
|
||||
parser.add_argument(
|
||||
'--project',
|
||||
metavar='<project>',
|
||||
help=_('Filter results by a project(admin only)')
|
||||
)
|
||||
return parser
|
||||
|
||||
def take_action(self, parsed_args):
|
||||
self.log.debug("take_action(%s)", parsed_args)
|
||||
data_protection_client = self.app.client_manager.data_protection
|
||||
all_projects = bool(parsed_args.project) or parsed_args.all_projects
|
||||
|
||||
search_opts = {
|
||||
'all_tenants': all_projects,
|
||||
'project_id': parsed_args.project,
|
||||
'name': parsed_args.name,
|
||||
'operation_type': parsed_args.operation_type,
|
||||
'trigger_id': parsed_args.trigger_id,
|
||||
'operation_definition': parsed_args.operation_definition,
|
||||
}
|
||||
|
||||
data = data_protection_client.scheduled_operations.list(
|
||||
search_opts=search_opts, marker=parsed_args.marker,
|
||||
limit=parsed_args.limit, sort=parsed_args.sort)
|
||||
|
||||
column_headers = ['Id', 'Name', 'OperationType', 'TriggerId',
|
||||
'OperationDefinition']
|
||||
|
||||
scheduled_operations = []
|
||||
for s in data:
|
||||
scheduled_operation = (s.id, s.name, s.operation_type,
|
||||
s.trigger_id, s.operation_definition)
|
||||
scheduled_operations.append(scheduled_operation)
|
||||
|
||||
return (column_headers, scheduled_operations)
|
||||
|
||||
|
||||
class ShowScheduledOperation(command.ShowOne):
|
||||
_description = "Shows scheduled_operation details"
|
||||
|
||||
def get_parser(self, prog_name):
|
||||
parser = super(ShowScheduledOperation, self).get_parser(prog_name)
|
||||
parser.add_argument(
|
||||
'scheduledoperation',
|
||||
metavar="<scheduledoperation>",
|
||||
help=_('The UUID of the scheduledoperation.')
|
||||
)
|
||||
return parser
|
||||
|
||||
def take_action(self, parsed_args):
|
||||
client = self.app.client_manager.data_protection
|
||||
so = osc_utils.find_resource(client.scheduled_operations,
|
||||
parsed_args.scheduledoperation)
|
||||
|
||||
so._info.pop("links", None)
|
||||
return zip(*sorted(six.iteritems(so._info)))
|
||||
|
||||
|
||||
class CreateScheduledOperation(command.ShowOne):
|
||||
_description = "Creates a scheduled operation"
|
||||
|
||||
def get_parser(self, prog_name):
|
||||
parser = super(CreateScheduledOperation, self).get_parser(prog_name)
|
||||
parser.add_argument(
|
||||
'name',
|
||||
metavar='<name>',
|
||||
help=_('The name of the scheduled operation.')
|
||||
)
|
||||
parser.add_argument(
|
||||
'operation_type',
|
||||
metavar='<operation_type>',
|
||||
help=_('Operation Type of scheduled operation.')
|
||||
)
|
||||
parser.add_argument(
|
||||
'trigger_id',
|
||||
metavar='<trigger_id>',
|
||||
help=_('Trigger id of scheduled operation.')
|
||||
)
|
||||
parser.add_argument(
|
||||
'operation_definition',
|
||||
metavar='<key=value,key=value>',
|
||||
help=_('Operation definition of scheduled operation.')
|
||||
)
|
||||
return parser
|
||||
|
||||
def take_action(self, parsed_args):
|
||||
client = self.app.client_manager.data_protection
|
||||
if not uuidutils.is_uuid_like(parsed_args.trigger_id):
|
||||
raise exceptions.CommandError(
|
||||
"Invalid trigger id provided.")
|
||||
so = client.scheduled_operations.create(
|
||||
parsed_args.name, parsed_args.operation_type,
|
||||
parsed_args.trigger_id, parsed_args.operation_definition)
|
||||
|
||||
so._info.pop("links", None)
|
||||
return zip(*sorted(six.iteritems(so._info)))
|
||||
|
||||
|
||||
class DeleteScheduledOperation(command.Command):
|
||||
_description = "Delete scheduled operation"
|
||||
|
||||
def get_parser(self, prog_name):
|
||||
parser = super(DeleteScheduledOperation, self).get_parser(prog_name)
|
||||
parser.add_argument(
|
||||
'scheduledoperation',
|
||||
metavar='<scheduledoperation>',
|
||||
nargs="+",
|
||||
help=_('ID of scheduled operation.')
|
||||
)
|
||||
return parser
|
||||
|
||||
def take_action(self, parsed_args):
|
||||
client = self.app.client_manager.data_protection
|
||||
failure_count = 0
|
||||
for so_id in parsed_args.scheduledoperation:
|
||||
try:
|
||||
so = osc_utils.find_resource(client.scheduled_operations,
|
||||
so_id)
|
||||
client.scheduled_operations.delete(so.id)
|
||||
except exceptions.NotFound:
|
||||
failure_count += 1
|
||||
print("Failed to delete '%s'; scheduled operation "
|
||||
"not found" % so_id)
|
||||
if failure_count == len(parsed_args.scheduledoperation):
|
||||
raise exceptions.CommandError(
|
||||
"Unable to find and delete any of the "
|
||||
"specified scheduled operation.")
|
||||
167
karborclient/tests/unit/osc/v1/test_scheduledoperations.py
Normal file
167
karborclient/tests/unit/osc/v1/test_scheduledoperations.py
Normal file
@@ -0,0 +1,167 @@
|
||||
# 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 karborclient.osc.v1 import scheduled_operations as osc_so
|
||||
from karborclient.tests.unit.osc.v1 import fakes
|
||||
from karborclient.v1 import scheduled_operations
|
||||
|
||||
|
||||
SCHEDULEDOPERATION_INFO = {
|
||||
"id": "1a2c0c3d-f402-4cd8-b5db-82e85cb51fad",
|
||||
"name": "My scheduled operation",
|
||||
"description": "It will run everyday",
|
||||
"operation_type": "protect",
|
||||
"trigger_id": "23902b02-5666-4ee6-8dfe-962ac09c3995",
|
||||
"operation_definition": {
|
||||
"provider_id": "2a9ce1f3-cc1a-4516-9435-0ebb13caa399",
|
||||
"plan_id": "2a9ce1f3-cc1a-4516-9435-0ebb13caa398"
|
||||
},
|
||||
"enabled": 1
|
||||
}
|
||||
|
||||
|
||||
class TestScheduledOperations(fakes.TestDataProtection):
|
||||
def setUp(self):
|
||||
super(TestScheduledOperations, self).setUp()
|
||||
self.so_mock = self.app.client_manager.data_protection.\
|
||||
scheduled_operations
|
||||
self.so_mock.reset_mock()
|
||||
|
||||
|
||||
class TestListScheduledOperations(TestScheduledOperations):
|
||||
def setUp(self):
|
||||
super(TestListScheduledOperations, self).setUp()
|
||||
self.so_mock.list.return_value = \
|
||||
[scheduled_operations.ScheduledOperation(None,
|
||||
SCHEDULEDOPERATION_INFO)]
|
||||
|
||||
# Command to test
|
||||
self.cmd = osc_so.ListScheduledOperations(self.app, None)
|
||||
|
||||
def test_scheduled_operations_list(self):
|
||||
arglist = ['--name', 'My scheduled operation']
|
||||
verifylist = [('name', 'My scheduled operation')]
|
||||
|
||||
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||
|
||||
columns, data = self.cmd.take_action(parsed_args)
|
||||
|
||||
# Check that columns are correct
|
||||
expected_columns = (
|
||||
['Id', 'Name', 'OperationType', 'TriggerId',
|
||||
'OperationDefinition'])
|
||||
self.assertEqual(expected_columns, columns)
|
||||
|
||||
# Check that data is correct
|
||||
expected_data = [("1a2c0c3d-f402-4cd8-b5db-82e85cb51fad",
|
||||
"My scheduled operation",
|
||||
"protect",
|
||||
"23902b02-5666-4ee6-8dfe-962ac09c3995",
|
||||
{
|
||||
"provider_id":
|
||||
"2a9ce1f3-cc1a-4516-9435-0ebb13caa399", # noqa
|
||||
"plan_id":
|
||||
"2a9ce1f3-cc1a-4516-9435-0ebb13caa398"
|
||||
})]
|
||||
self.assertEqual(expected_data, data)
|
||||
|
||||
|
||||
class TestCreateScheduledOperation(TestScheduledOperations):
|
||||
def setUp(self):
|
||||
super(TestCreateScheduledOperation, self).setUp()
|
||||
self.so_mock.create.return_value = scheduled_operations.\
|
||||
ScheduledOperation(None, SCHEDULEDOPERATION_INFO)
|
||||
# Command to test
|
||||
self.cmd = osc_so.CreateScheduledOperation(self.app, None)
|
||||
|
||||
def test_scheduled_operation_create(self):
|
||||
arglist = ['My scheduled operation',
|
||||
'protect',
|
||||
"23902b02-5666-4ee6-8dfe-962ac09c3995",
|
||||
"'provider_id=2a9ce1f3-cc1a-4516-9435-0ebb13caa399,"
|
||||
"plan_id=2a9ce1f3-cc1a-4516-9435-0ebb13caa398'"]
|
||||
verifylist = [('name', 'My scheduled operation'),
|
||||
('operation_type', 'protect'),
|
||||
('trigger_id', "23902b02-5666-4ee6-8dfe-962ac09c3995"),
|
||||
('operation_definition',
|
||||
"'provider_id=2a9ce1f3-cc1a-4516-9435-0ebb13caa399,"
|
||||
"plan_id=2a9ce1f3-cc1a-4516-9435-0ebb13caa398'")]
|
||||
|
||||
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||
|
||||
self.cmd.take_action(parsed_args)
|
||||
|
||||
# Check that correct arguments were passed
|
||||
self.so_mock.create.assert_called_once_with(
|
||||
'My scheduled operation',
|
||||
'protect',
|
||||
'23902b02-5666-4ee6-8dfe-962ac09c3995',
|
||||
"'provider_id=2a9ce1f3-cc1a-4516-9435-0ebb13caa399,"
|
||||
"plan_id=2a9ce1f3-cc1a-4516-9435-0ebb13caa398'")
|
||||
|
||||
|
||||
class TestDeleteScheduledOperation(TestScheduledOperations):
|
||||
def setUp(self):
|
||||
super(TestDeleteScheduledOperation, self).setUp()
|
||||
self.so_mock.get.return_value = scheduled_operations.\
|
||||
ScheduledOperation(None, SCHEDULEDOPERATION_INFO)
|
||||
# Command to test
|
||||
self.cmd = osc_so.DeleteScheduledOperation(self.app, None)
|
||||
|
||||
def test_scheduled_operation_delete(self):
|
||||
arglist = ['1a2c0c3d-f402-4cd8-b5db-82e85cb51fad']
|
||||
verifylist = [('scheduledoperation',
|
||||
['1a2c0c3d-f402-4cd8-b5db-82e85cb51fad'])]
|
||||
|
||||
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||
|
||||
self.cmd.take_action(parsed_args)
|
||||
|
||||
# Check that correct arguments were passed
|
||||
self.so_mock.delete.assert_called_once_with(
|
||||
'1a2c0c3d-f402-4cd8-b5db-82e85cb51fad')
|
||||
|
||||
|
||||
class TestShowScheduledOperation(TestScheduledOperations):
|
||||
def setUp(self):
|
||||
super(TestShowScheduledOperation, self).setUp()
|
||||
self.so_mock.get.return_value = scheduled_operations.\
|
||||
ScheduledOperation(None, SCHEDULEDOPERATION_INFO)
|
||||
|
||||
# Command to test
|
||||
self.cmd = osc_so.ShowScheduledOperation(self.app, None)
|
||||
|
||||
def test_scheduled_operation_show(self):
|
||||
arglist = ['1a2c0c3d-f402-4cd8-b5db-82e85cb51fad']
|
||||
verifylist = [('scheduledoperation',
|
||||
'1a2c0c3d-f402-4cd8-b5db-82e85cb51fad')]
|
||||
|
||||
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||
|
||||
columns, data = self.cmd.take_action(parsed_args)
|
||||
|
||||
# Check that columns are correct
|
||||
expected_columns = (
|
||||
'description', 'enabled', 'id', 'name', 'operation_definition',
|
||||
'operation_type', 'trigger_id')
|
||||
self.assertEqual(expected_columns, columns)
|
||||
|
||||
# Check that data is correct
|
||||
self.assertEqual(SCHEDULEDOPERATION_INFO['description'], data[0])
|
||||
self.assertEqual(SCHEDULEDOPERATION_INFO['enabled'], data[1])
|
||||
self.assertEqual(SCHEDULEDOPERATION_INFO['id'], data[2])
|
||||
self.assertEqual(SCHEDULEDOPERATION_INFO['name'], data[3])
|
||||
self.assertEqual(SCHEDULEDOPERATION_INFO['operation_definition'],
|
||||
data[4])
|
||||
self.assertEqual(SCHEDULEDOPERATION_INFO['operation_type'], data[5])
|
||||
self.assertEqual(SCHEDULEDOPERATION_INFO['trigger_id'], data[6])
|
||||
@@ -57,6 +57,10 @@ openstack.data_protection.v1 =
|
||||
data_protection_checkpoint_show = karborclient.osc.v1.checkpoints:ShowCheckpoint
|
||||
data_protection_checkpoint_create = karborclient.osc.v1.checkpoints:CreateCheckpoint
|
||||
data_protection_checkpoint_delete = karborclient.osc.v1.checkpoints:DeleteCheckpoint
|
||||
data_protection_scheduledoperation_list = karborclient.osc.v1.scheduled_operations:ListScheduledOperations
|
||||
data_protection_scheduledoperation_show = karborclient.osc.v1.scheduled_operations:ShowScheduledOperation
|
||||
data_protection_scheduledoperation_create = karborclient.osc.v1.scheduled_operations:CreateScheduledOperation
|
||||
data_protection_scheduledoperation_delete = karborclient.osc.v1.scheduled_operations:DeleteScheduledOperation
|
||||
|
||||
[compile_catalog]
|
||||
directory = karborclient/locale
|
||||
|
||||
Reference in New Issue
Block a user