OSC Network Meter
Implement Neutron feature of Metering into OpenStack Client. Meter Rules will be implemented in seperate patchset. Partially Implements: blueprint neutron-client-metering Change-Id: Ie82d42759504cbdb1c991c5183c1f0adf59e60fe
This commit is contained in:
parent
0948aa6aeb
commit
0fb1378c6c
91
doc/source/command-objects/meter.rst
Normal file
91
doc/source/command-objects/meter.rst
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
=============
|
||||||
|
network meter
|
||||||
|
=============
|
||||||
|
|
||||||
|
A **network meter** allows operators to measure
|
||||||
|
traffic for a specific IP range. The following commands
|
||||||
|
are specific to the L3 metering extension.
|
||||||
|
|
||||||
|
Network v2
|
||||||
|
|
||||||
|
network meter create
|
||||||
|
--------------------
|
||||||
|
|
||||||
|
Create network meter
|
||||||
|
|
||||||
|
.. program:: network meter create
|
||||||
|
.. code:: bash
|
||||||
|
|
||||||
|
openstack network meter create
|
||||||
|
[--project <project> [--project-domain <project-domain>]]
|
||||||
|
[--description <description>]
|
||||||
|
[--share | --no-share]
|
||||||
|
<name>
|
||||||
|
|
||||||
|
.. option:: --project <project>
|
||||||
|
|
||||||
|
Owner's project (name of ID)
|
||||||
|
|
||||||
|
*Network version 2 only*
|
||||||
|
|
||||||
|
.. option:: --description <description>
|
||||||
|
|
||||||
|
Description of meter
|
||||||
|
|
||||||
|
*Network version 2 only*
|
||||||
|
|
||||||
|
.. option:: --share
|
||||||
|
|
||||||
|
Share the meter between projects
|
||||||
|
|
||||||
|
.. option:: --no-share
|
||||||
|
|
||||||
|
Do not share the meter between projects (Default)
|
||||||
|
|
||||||
|
.. _network_meter_create:
|
||||||
|
.. describe:: <name>
|
||||||
|
|
||||||
|
New meter name
|
||||||
|
|
||||||
|
network meter delete
|
||||||
|
--------------------
|
||||||
|
|
||||||
|
Delete network meter(s)
|
||||||
|
|
||||||
|
.. program:: network meter delete
|
||||||
|
.. code:: bash
|
||||||
|
|
||||||
|
openstack network meter delete
|
||||||
|
<meter> [<meter> ...]
|
||||||
|
|
||||||
|
.. _network_meter_delete:
|
||||||
|
.. describe:: <meter>
|
||||||
|
|
||||||
|
Meter(s) to delete (name or ID)
|
||||||
|
|
||||||
|
network meter list
|
||||||
|
------------------
|
||||||
|
|
||||||
|
List network meters
|
||||||
|
|
||||||
|
.. program:: network meter list
|
||||||
|
.. code:: bash
|
||||||
|
|
||||||
|
openstack network meter list
|
||||||
|
|
||||||
|
|
||||||
|
network meter show
|
||||||
|
------------------
|
||||||
|
|
||||||
|
Show network meter
|
||||||
|
|
||||||
|
.. program:: network meter show
|
||||||
|
.. code:: bash
|
||||||
|
|
||||||
|
openstack network meter show
|
||||||
|
<meter>
|
||||||
|
|
||||||
|
.. _network_meter_show:
|
||||||
|
.. describe:: <meter>
|
||||||
|
|
||||||
|
Meter to display (name or ID)
|
@ -111,6 +111,7 @@ referring to both Compute and Volume quotas.
|
|||||||
* ``module``: (**Internal**) - installed Python modules in the OSC process
|
* ``module``: (**Internal**) - installed Python modules in the OSC process
|
||||||
* ``network``: (**Compute**, **Network**) - a virtual network for connecting servers and other resources
|
* ``network``: (**Compute**, **Network**) - a virtual network for connecting servers and other resources
|
||||||
* ``network agent``: (**Network**) - A network agent is an agent that handles various tasks used to implement virtual networks
|
* ``network agent``: (**Network**) - A network agent is an agent that handles various tasks used to implement virtual networks
|
||||||
|
* ``network meter``: (**Network**) - allow traffic metering in a network
|
||||||
* ``network rbac``: (**Network**) - an RBAC policy for network resources
|
* ``network rbac``: (**Network**) - an RBAC policy for network resources
|
||||||
* ``network qos policy``: (**Network**) - a QoS policy for network resources
|
* ``network qos policy``: (**Network**) - a QoS policy for network resources
|
||||||
* ``network qos rule type``: (**Network**) - list of QoS available rule types
|
* ``network qos rule type``: (**Network**) - list of QoS available rule types
|
||||||
|
190
openstackclient/network/v2/meter.py
Normal file
190
openstackclient/network/v2/meter.py
Normal file
@ -0,0 +1,190 @@
|
|||||||
|
# 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.
|
||||||
|
#
|
||||||
|
|
||||||
|
"""Metering Label Implementations"""
|
||||||
|
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from osc_lib.command import command
|
||||||
|
from osc_lib import exceptions
|
||||||
|
from osc_lib import utils
|
||||||
|
|
||||||
|
from openstackclient.i18n import _
|
||||||
|
from openstackclient.identity import common as identity_common
|
||||||
|
from openstackclient.network import sdk_utils
|
||||||
|
|
||||||
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
def _get_columns(item):
|
||||||
|
column_map = {
|
||||||
|
'is_shared': 'shared',
|
||||||
|
'tenant_id': 'project_id',
|
||||||
|
}
|
||||||
|
return sdk_utils.get_osc_show_columns_for_sdk_resource(item, column_map)
|
||||||
|
|
||||||
|
|
||||||
|
def _get_attrs(client_manager, parsed_args):
|
||||||
|
attrs = {}
|
||||||
|
|
||||||
|
if parsed_args.description is not None:
|
||||||
|
attrs['description'] = parsed_args.description
|
||||||
|
if parsed_args.project is not None and 'project' in parsed_args:
|
||||||
|
identity_client = client_manager.identity
|
||||||
|
project_id = identity_common.find_project(
|
||||||
|
identity_client,
|
||||||
|
parsed_args.project,
|
||||||
|
parsed_args.project_domain,
|
||||||
|
).id
|
||||||
|
attrs['tenant_id'] = project_id
|
||||||
|
if parsed_args.share:
|
||||||
|
attrs['shared'] = True
|
||||||
|
if parsed_args.no_share:
|
||||||
|
attrs['shared'] = False
|
||||||
|
if parsed_args.name is not None:
|
||||||
|
attrs['name'] = parsed_args.name
|
||||||
|
|
||||||
|
return attrs
|
||||||
|
|
||||||
|
|
||||||
|
# TODO(ankur-gupta-f): Use the SDK resource mapped attribute names once the
|
||||||
|
# OSC minimum requirements include SDK 1.0.
|
||||||
|
class CreateMeter(command.ShowOne):
|
||||||
|
_description = _("Create network meter")
|
||||||
|
|
||||||
|
def get_parser(self, prog_name):
|
||||||
|
parser = super(CreateMeter, self).get_parser(prog_name)
|
||||||
|
|
||||||
|
parser.add_argument(
|
||||||
|
'--description',
|
||||||
|
metavar='<description>',
|
||||||
|
help=_("Create description for meter")
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
'--project',
|
||||||
|
metavar='<project>',
|
||||||
|
help=_("Owner's project (name or ID)")
|
||||||
|
)
|
||||||
|
|
||||||
|
identity_common.add_project_domain_option_to_parser(parser)
|
||||||
|
share_group = parser.add_mutually_exclusive_group()
|
||||||
|
share_group.add_argument(
|
||||||
|
'--share',
|
||||||
|
action='store_true',
|
||||||
|
default=None,
|
||||||
|
help=_("Share meter between projects")
|
||||||
|
)
|
||||||
|
share_group.add_argument(
|
||||||
|
'--no-share',
|
||||||
|
action='store_true',
|
||||||
|
help=_("Do not share meter between projects")
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
'name',
|
||||||
|
metavar='<name>',
|
||||||
|
help=_('Name of meter'),
|
||||||
|
)
|
||||||
|
|
||||||
|
return parser
|
||||||
|
|
||||||
|
def take_action(self, parsed_args):
|
||||||
|
client = self.app.client_manager.network
|
||||||
|
attrs = _get_attrs(self.app.client_manager, parsed_args)
|
||||||
|
obj = client.create_metering_label(**attrs)
|
||||||
|
display_columns, columns = _get_columns(obj)
|
||||||
|
data = utils.get_item_properties(obj, columns, formatters={})
|
||||||
|
|
||||||
|
return (display_columns, data)
|
||||||
|
|
||||||
|
|
||||||
|
# TODO(ankur-gupta-f): Use the SDK resource mapped attribute names once the
|
||||||
|
# OSC minimum requirements include SDK 1.0.
|
||||||
|
class DeleteMeter(command.Command):
|
||||||
|
_description = _("Delete network meter")
|
||||||
|
|
||||||
|
def get_parser(self, prog_name):
|
||||||
|
parser = super(DeleteMeter, self).get_parser(prog_name)
|
||||||
|
|
||||||
|
parser.add_argument(
|
||||||
|
'meter',
|
||||||
|
metavar='<meter>',
|
||||||
|
nargs='+',
|
||||||
|
help=_('Meter to delete (name or ID)')
|
||||||
|
)
|
||||||
|
return parser
|
||||||
|
|
||||||
|
def take_action(self, parsed_args):
|
||||||
|
client = self.app.client_manager.network
|
||||||
|
result = 0
|
||||||
|
|
||||||
|
for meter in parsed_args.meter:
|
||||||
|
try:
|
||||||
|
obj = client.find_metering_label(meter, ignore_missing=False)
|
||||||
|
client.delete_metering_label(obj)
|
||||||
|
except Exception as e:
|
||||||
|
result += 1
|
||||||
|
LOG.error(_("Failed to delete meter with "
|
||||||
|
"ID '%(meter)s': %(e)s"),
|
||||||
|
{"meter": meter, "e": e})
|
||||||
|
if result > 0:
|
||||||
|
total = len(parsed_args.meter)
|
||||||
|
msg = (_("%(result)s of %(total)s meters failed "
|
||||||
|
"to delete.") % {"result": result, "total": total})
|
||||||
|
raise exceptions.CommandError(msg)
|
||||||
|
|
||||||
|
|
||||||
|
class ListMeter(command.Lister):
|
||||||
|
_description = _("List network meters")
|
||||||
|
|
||||||
|
def take_action(self, parsed_args):
|
||||||
|
client = self.app.client_manager.network
|
||||||
|
|
||||||
|
columns = (
|
||||||
|
'id',
|
||||||
|
'name',
|
||||||
|
'description',
|
||||||
|
'shared',
|
||||||
|
)
|
||||||
|
column_headers = (
|
||||||
|
'ID',
|
||||||
|
'Name',
|
||||||
|
'Description',
|
||||||
|
'Shared',
|
||||||
|
)
|
||||||
|
|
||||||
|
data = client.metering_labels()
|
||||||
|
return (column_headers,
|
||||||
|
(utils.get_item_properties(
|
||||||
|
s, columns,
|
||||||
|
) for s in data))
|
||||||
|
|
||||||
|
|
||||||
|
class ShowMeter(command.ShowOne):
|
||||||
|
_description = _("Show network meter")
|
||||||
|
|
||||||
|
def get_parser(self, prog_name):
|
||||||
|
parser = super(ShowMeter, self).get_parser(prog_name)
|
||||||
|
parser.add_argument(
|
||||||
|
'meter',
|
||||||
|
metavar='<meter>',
|
||||||
|
help=_('Meter to display (name or ID)')
|
||||||
|
)
|
||||||
|
return parser
|
||||||
|
|
||||||
|
def take_action(self, parsed_args):
|
||||||
|
client = self.app.client_manager.network
|
||||||
|
obj = client.find_metering_label(parsed_args.meter,
|
||||||
|
ignore_missing=False)
|
||||||
|
display_columns, columns = _get_columns(obj)
|
||||||
|
data = utils.get_item_properties(obj, columns)
|
||||||
|
return display_columns, data
|
102
openstackclient/tests/functional/network/v2/test_meter.py
Normal file
102
openstackclient/tests/functional/network/v2/test_meter.py
Normal file
@ -0,0 +1,102 @@
|
|||||||
|
# Copyright (c) 2016, Intel Corporation.
|
||||||
|
# All Rights Reserved.
|
||||||
|
#
|
||||||
|
# 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 re
|
||||||
|
import uuid
|
||||||
|
|
||||||
|
from openstackclient.tests.functional import base
|
||||||
|
|
||||||
|
|
||||||
|
class TestMeter(base.TestCase):
|
||||||
|
"""Functional tests for network meter."""
|
||||||
|
|
||||||
|
# NOTE(dtroyer): Do not normalize the setup and teardown of the resource
|
||||||
|
# creation and deletion. Little is gained when each test
|
||||||
|
# has its own needs and there are collisions when running
|
||||||
|
# tests in parallel.
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def setUpClass(cls):
|
||||||
|
# Set up some regex for matching below
|
||||||
|
cls.re_name = re.compile("name\s+\|\s+([^|]+?)\s+\|")
|
||||||
|
cls.re_shared = re.compile("shared\s+\|\s+(\S+)")
|
||||||
|
cls.re_description = re.compile("description\s+\|\s+([^|]+?)\s+\|")
|
||||||
|
|
||||||
|
def test_meter_delete(self):
|
||||||
|
"""Test create, delete multiple"""
|
||||||
|
name1 = uuid.uuid4().hex
|
||||||
|
name2 = uuid.uuid4().hex
|
||||||
|
|
||||||
|
raw_output = self.openstack(
|
||||||
|
'network meter create ' + name1,
|
||||||
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
name1,
|
||||||
|
re.search(self.re_name, raw_output).group(1),
|
||||||
|
)
|
||||||
|
# Check if default shared values
|
||||||
|
self.assertEqual(
|
||||||
|
'False',
|
||||||
|
re.search(self.re_shared, raw_output).group(1)
|
||||||
|
)
|
||||||
|
|
||||||
|
raw_output = self.openstack(
|
||||||
|
'network meter create ' + name2,
|
||||||
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
name2,
|
||||||
|
re.search(self.re_name, raw_output).group(1),
|
||||||
|
)
|
||||||
|
|
||||||
|
raw_output = self.openstack(
|
||||||
|
'network meter delete ' + name1 + ' ' + name2,
|
||||||
|
)
|
||||||
|
self.assertOutput('', raw_output)
|
||||||
|
|
||||||
|
def test_meter_list(self):
|
||||||
|
"""Test create, list filters, delete"""
|
||||||
|
name1 = uuid.uuid4().hex
|
||||||
|
raw_output = self.openstack(
|
||||||
|
'network meter create --description Test1 --share ' + name1,
|
||||||
|
)
|
||||||
|
self.addCleanup(self.openstack, 'network meter delete ' + name1)
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
'Test1',
|
||||||
|
re.search(self.re_description, raw_output).group(1),
|
||||||
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
'True',
|
||||||
|
re.search(self.re_shared, raw_output).group(1),
|
||||||
|
)
|
||||||
|
|
||||||
|
name2 = uuid.uuid4().hex
|
||||||
|
raw_output = self.openstack(
|
||||||
|
'network meter create --description Test2 --no-share ' + name2,
|
||||||
|
)
|
||||||
|
self.addCleanup(self.openstack, 'network meter delete ' + name2)
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
'Test2',
|
||||||
|
re.search(self.re_description, raw_output).group(1),
|
||||||
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
'False',
|
||||||
|
re.search(self.re_shared, raw_output).group(1),
|
||||||
|
)
|
||||||
|
|
||||||
|
raw_output = self.openstack('network meter list')
|
||||||
|
self.assertIsNotNone(re.search(name1 + "\s+\|\s+Test1", raw_output))
|
||||||
|
self.assertIsNotNone(re.search(name2 + "\s+\|\s+Test2", raw_output))
|
@ -1258,6 +1258,51 @@ class FakeFloatingIP(object):
|
|||||||
return mock.Mock(side_effect=floating_ips)
|
return mock.Mock(side_effect=floating_ips)
|
||||||
|
|
||||||
|
|
||||||
|
class FakeNetworkMeter(object):
|
||||||
|
"""Fake network meter"""
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def create_one_meter(attrs=None):
|
||||||
|
"""Create metering pool"""
|
||||||
|
attrs = attrs or {}
|
||||||
|
|
||||||
|
meter_attrs = {
|
||||||
|
'id': 'meter-id-' + uuid.uuid4().hex,
|
||||||
|
'name': 'meter-name-' + uuid.uuid4().hex,
|
||||||
|
'description': 'meter-description-' + uuid.uuid4().hex,
|
||||||
|
'tenant_id': 'project-id-' + uuid.uuid4().hex,
|
||||||
|
'shared': False
|
||||||
|
}
|
||||||
|
|
||||||
|
meter_attrs.update(attrs)
|
||||||
|
|
||||||
|
meter = fakes.FakeResource(
|
||||||
|
info=copy.deepcopy(meter_attrs),
|
||||||
|
loaded=True)
|
||||||
|
|
||||||
|
meter.project_id = meter_attrs['tenant_id']
|
||||||
|
|
||||||
|
return meter
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def create_meter(attrs=None, count=2):
|
||||||
|
"""Create multiple meters"""
|
||||||
|
|
||||||
|
meters = []
|
||||||
|
for i in range(0, count):
|
||||||
|
meters.append(FakeNetworkMeter.
|
||||||
|
create_one_meter(attrs))
|
||||||
|
return meters
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_meter(meter=None, count=2):
|
||||||
|
"""Get a list of meters"""
|
||||||
|
if meter is None:
|
||||||
|
meter = (FakeNetworkMeter.
|
||||||
|
create_meter(count))
|
||||||
|
return mock.Mock(side_effect=meter)
|
||||||
|
|
||||||
|
|
||||||
class FakeSubnetPool(object):
|
class FakeSubnetPool(object):
|
||||||
"""Fake one or more subnet pools."""
|
"""Fake one or more subnet pools."""
|
||||||
|
|
||||||
|
304
openstackclient/tests/unit/network/v2/test_meter.py
Normal file
304
openstackclient/tests/unit/network/v2/test_meter.py
Normal file
@ -0,0 +1,304 @@
|
|||||||
|
# Copyright (c) 2016, Intel Corporation.
|
||||||
|
# All Rights Reserved.
|
||||||
|
#
|
||||||
|
# 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 mock import call
|
||||||
|
|
||||||
|
from osc_lib import exceptions
|
||||||
|
|
||||||
|
from openstackclient.network.v2 import meter
|
||||||
|
from openstackclient.tests.unit.identity.v3 import fakes as identity_fakes_v3
|
||||||
|
from openstackclient.tests.unit.network.v2 import fakes as network_fakes
|
||||||
|
from openstackclient.tests.unit import utils as tests_utils
|
||||||
|
|
||||||
|
|
||||||
|
class TestMeter(network_fakes.TestNetworkV2):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestMeter, self).setUp()
|
||||||
|
self.network = self.app.client_manager.network
|
||||||
|
self.projects_mock = self.app.client_manager.identity.projects
|
||||||
|
self.domains_mock = self.app.client_manager.identity.domains
|
||||||
|
|
||||||
|
|
||||||
|
class TestCreateMeter(TestMeter):
|
||||||
|
project = identity_fakes_v3.FakeProject.create_one_project()
|
||||||
|
domain = identity_fakes_v3.FakeDomain.create_one_domain()
|
||||||
|
|
||||||
|
new_meter = (
|
||||||
|
network_fakes.FakeNetworkMeter.
|
||||||
|
create_one_meter()
|
||||||
|
)
|
||||||
|
columns = (
|
||||||
|
'description',
|
||||||
|
'id',
|
||||||
|
'name',
|
||||||
|
'project_id',
|
||||||
|
'shared',
|
||||||
|
)
|
||||||
|
|
||||||
|
data = (
|
||||||
|
new_meter.description,
|
||||||
|
new_meter.id,
|
||||||
|
new_meter.name,
|
||||||
|
new_meter.project_id,
|
||||||
|
new_meter.shared,
|
||||||
|
)
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestCreateMeter, self).setUp()
|
||||||
|
self.network.create_metering_label = mock.Mock(
|
||||||
|
return_value=self.new_meter)
|
||||||
|
self.projects_mock.get.return_value = self.project
|
||||||
|
self.cmd = meter.CreateMeter(self.app, self.namespace)
|
||||||
|
|
||||||
|
def test_create_no_options(self):
|
||||||
|
arglist = []
|
||||||
|
verifylist = []
|
||||||
|
|
||||||
|
self.assertRaises(tests_utils.ParserException, self.check_parser,
|
||||||
|
self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
def test_create_default_options(self):
|
||||||
|
arglist = [
|
||||||
|
self.new_meter.name,
|
||||||
|
]
|
||||||
|
|
||||||
|
verifylist = [
|
||||||
|
('name', self.new_meter.name),
|
||||||
|
]
|
||||||
|
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
columns, data = (self.cmd.take_action(parsed_args))
|
||||||
|
|
||||||
|
self.network.create_metering_label.assert_called_once_with(
|
||||||
|
**{'name': self.new_meter.name}
|
||||||
|
)
|
||||||
|
self.assertEqual(self.columns, columns)
|
||||||
|
self.assertEqual(self.data, data)
|
||||||
|
|
||||||
|
def test_create_all_options(self):
|
||||||
|
arglist = [
|
||||||
|
"--description", self.new_meter.description,
|
||||||
|
"--project", self.new_meter.project_id,
|
||||||
|
"--project-domain", self.domain.name,
|
||||||
|
"--share",
|
||||||
|
self.new_meter.name,
|
||||||
|
]
|
||||||
|
|
||||||
|
verifylist = [
|
||||||
|
('description', self.new_meter.description),
|
||||||
|
('name', self.new_meter.name),
|
||||||
|
('project', self.new_meter.project_id),
|
||||||
|
('project_domain', self.domain.name),
|
||||||
|
('share', True),
|
||||||
|
]
|
||||||
|
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
columns, data = (self.cmd.take_action(parsed_args))
|
||||||
|
|
||||||
|
self.network.create_metering_label.assert_called_once_with(
|
||||||
|
**{'description': self.new_meter.description,
|
||||||
|
'name': self.new_meter.name,
|
||||||
|
'tenant_id': self.project.id,
|
||||||
|
'shared': True, }
|
||||||
|
)
|
||||||
|
self.assertEqual(self.columns, columns)
|
||||||
|
self.assertEqual(self.data, data)
|
||||||
|
|
||||||
|
|
||||||
|
class TestDeleteMeter(TestMeter):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestDeleteMeter, self).setUp()
|
||||||
|
|
||||||
|
self.meter_list = \
|
||||||
|
network_fakes.FakeNetworkMeter.create_meter(count=2)
|
||||||
|
|
||||||
|
self.network.delete_metering_label = mock.Mock(return_value=None)
|
||||||
|
|
||||||
|
self.network.find_metering_label = network_fakes \
|
||||||
|
.FakeNetworkMeter.get_meter(
|
||||||
|
meter=self.meter_list
|
||||||
|
)
|
||||||
|
|
||||||
|
self.cmd = meter.DeleteMeter(self.app, self.namespace)
|
||||||
|
|
||||||
|
def test_delete_one_meter(self):
|
||||||
|
arglist = [
|
||||||
|
self.meter_list[0].name,
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('meter', [self.meter_list[0].name]),
|
||||||
|
]
|
||||||
|
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
result = self.cmd.take_action(parsed_args)
|
||||||
|
|
||||||
|
self.network.delete_metering_label.assert_called_once_with(
|
||||||
|
self.meter_list[0]
|
||||||
|
)
|
||||||
|
self.assertIsNone(result)
|
||||||
|
|
||||||
|
def test_delete_multiple_meters(self):
|
||||||
|
arglist = []
|
||||||
|
for n in self.meter_list:
|
||||||
|
arglist.append(n.id)
|
||||||
|
verifylist = [
|
||||||
|
('meter', arglist),
|
||||||
|
]
|
||||||
|
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
result = self.cmd.take_action(parsed_args)
|
||||||
|
|
||||||
|
calls = []
|
||||||
|
for n in self.meter_list:
|
||||||
|
calls.append(call(n))
|
||||||
|
self.network.delete_metering_label.assert_has_calls(calls)
|
||||||
|
self.assertIsNone(result)
|
||||||
|
|
||||||
|
def test_delete_multiple_meter_exception(self):
|
||||||
|
arglist = [
|
||||||
|
self.meter_list[0].id,
|
||||||
|
'xxxx-yyyy-zzzz',
|
||||||
|
self.meter_list[1].id,
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('meter', arglist),
|
||||||
|
]
|
||||||
|
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
return_find = [
|
||||||
|
self.meter_list[0],
|
||||||
|
exceptions.NotFound('404'),
|
||||||
|
self.meter_list[1],
|
||||||
|
]
|
||||||
|
self.network.find_meter = mock.Mock(side_effect=return_find)
|
||||||
|
|
||||||
|
ret_delete = [
|
||||||
|
None,
|
||||||
|
exceptions.NotFound('404'),
|
||||||
|
]
|
||||||
|
self.network.delete_metering_label = mock.Mock(side_effect=ret_delete)
|
||||||
|
|
||||||
|
self.assertRaises(exceptions.CommandError, self.cmd.take_action,
|
||||||
|
parsed_args)
|
||||||
|
|
||||||
|
calls = [
|
||||||
|
call(self.meter_list[0]),
|
||||||
|
call(self.meter_list[1]),
|
||||||
|
]
|
||||||
|
self.network.delete_metering_label.assert_has_calls(calls)
|
||||||
|
|
||||||
|
|
||||||
|
class TestListMeter(TestMeter):
|
||||||
|
|
||||||
|
meter_list = \
|
||||||
|
network_fakes.FakeNetworkMeter.create_meter(count=2)
|
||||||
|
|
||||||
|
columns = (
|
||||||
|
'ID',
|
||||||
|
'Name',
|
||||||
|
'Description',
|
||||||
|
'Shared',
|
||||||
|
)
|
||||||
|
|
||||||
|
data = []
|
||||||
|
|
||||||
|
for meters in meter_list:
|
||||||
|
data.append((
|
||||||
|
meters.id,
|
||||||
|
meters.name,
|
||||||
|
meters.description,
|
||||||
|
meters.shared,
|
||||||
|
))
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestListMeter, self).setUp()
|
||||||
|
|
||||||
|
self.network.metering_labels = mock.Mock(
|
||||||
|
return_value=self.meter_list
|
||||||
|
)
|
||||||
|
|
||||||
|
self.cmd = meter.ListMeter(self.app, self.namespace)
|
||||||
|
|
||||||
|
def test_meter_list(self):
|
||||||
|
arglist = []
|
||||||
|
verifylist = []
|
||||||
|
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
columns, data = self.cmd.take_action(parsed_args)
|
||||||
|
|
||||||
|
self.network.metering_labels.assert_called_with()
|
||||||
|
self.assertEqual(self.columns, columns)
|
||||||
|
self.assertEqual(self.data, list(data))
|
||||||
|
|
||||||
|
|
||||||
|
class TestShowMeter(TestMeter):
|
||||||
|
new_meter = (
|
||||||
|
network_fakes.FakeNetworkMeter.
|
||||||
|
create_one_meter()
|
||||||
|
)
|
||||||
|
columns = (
|
||||||
|
'description',
|
||||||
|
'id',
|
||||||
|
'name',
|
||||||
|
'project_id',
|
||||||
|
'shared',
|
||||||
|
)
|
||||||
|
|
||||||
|
data = (
|
||||||
|
new_meter.description,
|
||||||
|
new_meter.id,
|
||||||
|
new_meter.name,
|
||||||
|
new_meter.project_id,
|
||||||
|
new_meter.shared,
|
||||||
|
)
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestShowMeter, self).setUp()
|
||||||
|
|
||||||
|
self.cmd = meter.ShowMeter(self.app, self.namespace)
|
||||||
|
|
||||||
|
self.network.find_metering_label = \
|
||||||
|
mock.Mock(return_value=self.new_meter)
|
||||||
|
|
||||||
|
def test_show_no_options(self):
|
||||||
|
arglist = []
|
||||||
|
verifylist = []
|
||||||
|
|
||||||
|
self.assertRaises(tests_utils.ParserException, self.check_parser,
|
||||||
|
self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
def test_meter_show_option(self):
|
||||||
|
arglist = [
|
||||||
|
self.new_meter.name,
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('meter', self.new_meter.name),
|
||||||
|
]
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
columns, data = self.cmd.take_action(parsed_args)
|
||||||
|
|
||||||
|
self.network.find_metering_label.assert_called_with(
|
||||||
|
self.new_meter.name, ignore_missing=False
|
||||||
|
)
|
||||||
|
self.assertEqual(self.columns, columns)
|
||||||
|
self.assertEqual(self.data, data)
|
@ -0,0 +1,7 @@
|
|||||||
|
---
|
||||||
|
features:
|
||||||
|
- |
|
||||||
|
Add support for network metering commands:
|
||||||
|
``network meter create``, ``network meter delete``,
|
||||||
|
``network meter show``, ``network meter list``
|
||||||
|
[Blueprint :oscbp:`neutron-client-metering`]
|
@ -355,6 +355,11 @@ openstack.network.v2 =
|
|||||||
|
|
||||||
ip_floating_pool_list = openstackclient.network.v2.floating_ip_pool:ListIPFloatingPool
|
ip_floating_pool_list = openstackclient.network.v2.floating_ip_pool:ListIPFloatingPool
|
||||||
|
|
||||||
|
network_meter_create = openstackclient.network.v2.meter:CreateMeter
|
||||||
|
network_meter_delete = openstackclient.network.v2.meter:DeleteMeter
|
||||||
|
network_meter_list = openstackclient.network.v2.meter:ListMeter
|
||||||
|
network_meter_show = openstackclient.network.v2.meter:ShowMeter
|
||||||
|
|
||||||
network_agent_delete = openstackclient.network.v2.network_agent:DeleteNetworkAgent
|
network_agent_delete = openstackclient.network.v2.network_agent:DeleteNetworkAgent
|
||||||
network_agent_list = openstackclient.network.v2.network_agent:ListNetworkAgent
|
network_agent_list = openstackclient.network.v2.network_agent:ListNetworkAgent
|
||||||
network_agent_set = openstackclient.network.v2.network_agent:SetNetworkAgent
|
network_agent_set = openstackclient.network.v2.network_agent:SetNetworkAgent
|
||||||
|
Loading…
Reference in New Issue
Block a user