Add Category support for openstack CLI
Adding category commands to murano openstack-client $ openstack category list $ openstack category show <id> $ openstack category create <name> $ openstack category delete <id> [<id> ...] Partially implements: blueprint openstack-client-plugin-support Change-Id: I5623b2fd18508e795115a5e646eb44ad3e82b836
This commit is contained in:
parent
9a86110b6d
commit
becbdeae41
150
muranoclient/osc/v1/category.py
Normal file
150
muranoclient/osc/v1/category.py
Normal file
@ -0,0 +1,150 @@
|
||||
# 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.
|
||||
|
||||
"""Application-catalog v1 category action implementation"""
|
||||
|
||||
import textwrap
|
||||
|
||||
from cliff import lister
|
||||
from cliff import show
|
||||
from muranoclient.openstack.common.apiclient import exceptions
|
||||
from openstackclient.common import utils
|
||||
from oslo_log import log as logging
|
||||
|
||||
LOG = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class ListCategories(lister.Lister):
|
||||
"""List all available categories."""
|
||||
|
||||
def take_action(self, parsed_args=None):
|
||||
LOG.debug("take_action({0})".format(parsed_args))
|
||||
client = self.app.client_manager.application_catalog
|
||||
|
||||
if parsed_args is None:
|
||||
parsed_args = {}
|
||||
|
||||
data = client.categories.list()
|
||||
|
||||
fields = ["id", "name"]
|
||||
field_labels = ["ID", "Name"]
|
||||
|
||||
return (
|
||||
field_labels,
|
||||
list(utils.get_item_properties(
|
||||
s,
|
||||
fields,
|
||||
) for s in data)
|
||||
)
|
||||
|
||||
|
||||
class ShowCategory(show.ShowOne):
|
||||
"""Display category details."""
|
||||
|
||||
def get_parser(self, prog_name):
|
||||
parser = super(ShowCategory, self).get_parser(prog_name)
|
||||
parser.add_argument(
|
||||
"id",
|
||||
metavar="<ID>",
|
||||
help=("ID of a category(s) to show."),
|
||||
)
|
||||
|
||||
return parser
|
||||
|
||||
def take_action(self, parsed_args):
|
||||
LOG.debug("take_action({0})".format(parsed_args))
|
||||
client = self.app.client_manager.application_catalog
|
||||
|
||||
category = client.categories.get(parsed_args.id)
|
||||
to_display = dict(id=category.id,
|
||||
name=category.name,
|
||||
packages=', '.join(p['name']
|
||||
for p in category.packages))
|
||||
|
||||
to_display['packages'] = '\n'.join(textwrap.wrap(to_display['packages']
|
||||
or '', 55))
|
||||
|
||||
return self.dict2columns(to_display)
|
||||
|
||||
|
||||
class CreateCategory(lister.Lister):
|
||||
"""Create a category."""
|
||||
|
||||
def get_parser(self, prog_name):
|
||||
parser = super(CreateCategory, self).get_parser(prog_name)
|
||||
parser.add_argument(
|
||||
"name",
|
||||
metavar="<CATEGORY_NAME>",
|
||||
help=("Category name."),
|
||||
)
|
||||
|
||||
return parser
|
||||
|
||||
def take_action(self, parsed_args):
|
||||
LOG.debug("take_action({0})".format(parsed_args))
|
||||
client = self.app.client_manager.application_catalog
|
||||
|
||||
data = [client.categories.add({"name": parsed_args.name})]
|
||||
|
||||
fields = ["id", "name"]
|
||||
field_labels = ["ID", "Name"]
|
||||
|
||||
return (
|
||||
field_labels,
|
||||
list(utils.get_item_properties(
|
||||
s,
|
||||
fields,
|
||||
) for s in data)
|
||||
)
|
||||
|
||||
|
||||
class DeleteCategory(lister.Lister):
|
||||
"""Delete a category."""
|
||||
|
||||
def get_parser(self, prog_name):
|
||||
parser = super(DeleteCategory, self).get_parser(prog_name)
|
||||
parser.add_argument(
|
||||
"id",
|
||||
metavar="<ID>",
|
||||
nargs="+",
|
||||
help=("ID of a category(ies) to delete."),
|
||||
)
|
||||
|
||||
return parser
|
||||
|
||||
def take_action(self, parsed_args):
|
||||
LOG.debug("take_action({0})".format(parsed_args))
|
||||
client = self.app.client_manager.application_catalog
|
||||
|
||||
failure_count = 0
|
||||
for category_id in parsed_args.id:
|
||||
try:
|
||||
client.categories.delete(category_id)
|
||||
except Exception:
|
||||
failure_count += 1
|
||||
print("Failed to delete '{0}'; category not found".
|
||||
format(category_id))
|
||||
if failure_count == len(parsed_args.id):
|
||||
raise exceptions.CommandError("Unable to find and delete any of "
|
||||
"the specified categories.")
|
||||
data = client.categories.list()
|
||||
|
||||
fields = ["id", "name"]
|
||||
field_labels = ["ID", "Name"]
|
||||
|
||||
return (
|
||||
field_labels,
|
||||
list(utils.get_item_properties(
|
||||
s,
|
||||
fields,
|
||||
) for s in data)
|
||||
)
|
141
muranoclient/tests/unit/osc/v1/test_category.py
Normal file
141
muranoclient/tests/unit/osc/v1/test_category.py
Normal file
@ -0,0 +1,141 @@
|
||||
# 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 mock
|
||||
|
||||
from muranoclient.osc.v1 import category as osc_category
|
||||
from muranoclient.tests.unit.osc.v1 import fakes
|
||||
from muranoclient.v1 import categories as api_category
|
||||
|
||||
CATEGORY_INFO = {'id': 'xyz123',
|
||||
'name': 'fake1',
|
||||
'packages': [{'name': 'package1'}, {'name': 'package2'}]}
|
||||
|
||||
|
||||
class TestCategory(fakes.TestApplicationCatalog):
|
||||
def setUp(self):
|
||||
super(TestCategory, self).setUp()
|
||||
self.category_mock = self.app.client_manager.application_catalog.\
|
||||
categories
|
||||
self.category_mock.reset_mock()
|
||||
|
||||
|
||||
class TestListCategories(TestCategory):
|
||||
def setUp(self):
|
||||
super(TestListCategories, self).setUp()
|
||||
self.category_mock.list.return_value = [api_category.Category(None,
|
||||
CATEGORY_INFO)]
|
||||
|
||||
# Command to test
|
||||
self.cmd = osc_category.ListCategories(self.app, None)
|
||||
|
||||
@mock.patch('openstackclient.common.utils.get_item_properties')
|
||||
def test_category_list(self, mock_util):
|
||||
mock_util.return_value = ('xyz123', 'fake1')
|
||||
|
||||
columns, data = self.cmd.take_action(parsed_args=None)
|
||||
|
||||
# Check that columns are correct
|
||||
expected_columns = ['ID', 'Name']
|
||||
self.assertEqual(expected_columns, columns)
|
||||
|
||||
# Check that data is correct
|
||||
expected_data = [('xyz123', 'fake1')]
|
||||
self.assertEqual(expected_data, data)
|
||||
|
||||
|
||||
class TestShowCategory(TestCategory):
|
||||
def setUp(self):
|
||||
super(TestShowCategory, self).setUp()
|
||||
self.category_mock.get.return_value = api_category.\
|
||||
Category(None, CATEGORY_INFO)
|
||||
|
||||
# Command to test
|
||||
self.cmd = osc_category.ShowCategory(self.app, None)
|
||||
|
||||
@mock.patch('textwrap.wrap')
|
||||
def test_category_show(self, mock_wrap):
|
||||
arglist = ['xyz123']
|
||||
verifylist = [('id', 'xyz123')]
|
||||
|
||||
mock_wrap.return_value = ['package1, package2']
|
||||
|
||||
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', 'packages')
|
||||
self.assertEqual(expected_columns, columns)
|
||||
|
||||
# Check that data is correct
|
||||
expected_data = ('xyz123', 'fake1', 'package1, package2')
|
||||
self.assertEqual(expected_data, data)
|
||||
|
||||
|
||||
class TestCreateCategory(TestCategory):
|
||||
def setUp(self):
|
||||
super(TestCreateCategory, self).setUp()
|
||||
self.category_mock.add.return_value = [api_category.Category(None,
|
||||
CATEGORY_INFO)]
|
||||
|
||||
# Command to test
|
||||
self.cmd = osc_category.CreateCategory(self.app, None)
|
||||
|
||||
@mock.patch('openstackclient.common.utils.get_item_properties')
|
||||
def test_category_list(self, mock_util):
|
||||
arglist = ['fake1']
|
||||
verifylist = [('name', 'fake1')]
|
||||
|
||||
mock_util.return_value = ('xyz123', 'fake1')
|
||||
|
||||
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']
|
||||
self.assertEqual(expected_columns, columns)
|
||||
|
||||
# Check that data is correct
|
||||
expected_data = [('xyz123', 'fake1')]
|
||||
self.assertEqual(expected_data, data)
|
||||
|
||||
|
||||
class TestDeleteCategory(TestCategory):
|
||||
def setUp(self):
|
||||
super(TestDeleteCategory, self).setUp()
|
||||
self.category_mock.delete.return_value = None
|
||||
self.category_mock.list.return_value = [api_category.Category(None,
|
||||
CATEGORY_INFO)]
|
||||
|
||||
# Command to test
|
||||
self.cmd = osc_category.DeleteCategory(self.app, None)
|
||||
|
||||
@mock.patch('openstackclient.common.utils.get_item_properties')
|
||||
def test_category_list(self, mock_util):
|
||||
arglist = ['abc123', '123abc']
|
||||
verifylist = [('id', ['abc123', '123abc'])]
|
||||
|
||||
mock_util.return_value = ('xyz123', 'fake1')
|
||||
|
||||
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']
|
||||
self.assertEqual(expected_columns, columns)
|
||||
|
||||
# Check that data is correct
|
||||
expected_data = [('xyz123', 'fake1')]
|
||||
self.assertEqual(expected_data, data)
|
@ -0,0 +1,6 @@
|
||||
---
|
||||
features:
|
||||
- New OSC command ``category list``
|
||||
- New OSC command ``category show <id>``
|
||||
- New OSC command ``category create <name>``
|
||||
- New OSC command ``category delete <id> [<id> ...]``
|
@ -39,6 +39,11 @@ openstack.application_catalog.v1 =
|
||||
environment_rename = muranoclient.osc.v1.environment:RenameEnvironment
|
||||
environment_session_create = muranoclient.osc.v1.environment:EnvironmentSessionCreate
|
||||
|
||||
category_list = muranoclient.osc.v1.category:ListCategories
|
||||
category_show = muranoclient.osc.v1.category:ShowCategory
|
||||
category_create = muranoclient.osc.v1.category:CreateCategory
|
||||
category_delete = muranoclient.osc.v1.category:DeleteCategory
|
||||
|
||||
[global]
|
||||
setup-hooks =
|
||||
pbr.hooks.setup_hook
|
||||
|
Loading…
Reference in New Issue
Block a user