Merge "Add network availability for osc"

This commit is contained in:
Jenkins 2016-06-06 08:20:15 +00:00 committed by Gerrit Code Review
commit 9661354979
7 changed files with 345 additions and 4 deletions

View File

@ -24,12 +24,12 @@ number of allocated IP addresses from that pool.
.. option:: --ip-version {4,6} .. option:: --ip-version {4,6}
List IP availability for specific version List IP availability of given IP version networks
(Default is 4) (Default is 4)
.. option:: --project <project> .. option:: --project <project>
List IP availability for specific project List IP availability of given project
(name or ID) (name or ID)
ip availability show ip availability show
@ -57,5 +57,4 @@ subnet within the network as well.
.. _ip_availability_show-network .. _ip_availability_show-network
.. describe:: <network> .. describe:: <network>
Show network IP availability for specific Show IP availability for a specific network (name or ID)
network (name or ID)

View File

@ -96,6 +96,7 @@ referring to both Compute and Volume quotas.
* ``hypervisor stats``: (**Compute**) hypervisor statistics over all compute nodes * ``hypervisor stats``: (**Compute**) hypervisor statistics over all compute nodes
* ``identity provider``: (**Identity**) a source of users and authentication * ``identity provider``: (**Identity**) a source of users and authentication
* ``image``: (**Image**) a disk image * ``image``: (**Image**) a disk image
* ``ip availability``: (**Network**) - details of IP usage of a network
* ``ip fixed``: (**Compute**, **Network**) - an internal IP address assigned to a server * ``ip fixed``: (**Compute**, **Network**) - an internal IP address assigned to a server
* ``ip floating``: (**Compute**, **Network**) - a public IP address that can be mapped to a server * ``ip floating``: (**Compute**, **Network**) - a public IP address that can be mapped to a server
* ``ip floating pool``: (**Compute**, **Network**) - a pool of public IP addresses * ``ip floating pool``: (**Compute**, **Network**) - a pool of public IP addresses

View File

@ -0,0 +1,109 @@
# 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.
#
"""IP Availability Info implementations"""
from openstackclient.common import command
from openstackclient.common import utils
from openstackclient.i18n import _
from openstackclient.identity import common as identity_common
_formatters = {
'subnet_ip_availability': utils.format_list_of_dicts,
}
def _get_columns(item):
columns = list(item.keys())
if 'tenant_id' in columns:
columns.remove('tenant_id')
columns.append('project_id')
return tuple(sorted(columns))
class ListIPAvailability(command.Lister):
"""List IP availability for network"""
def get_parser(self, prog_name):
parser = super(ListIPAvailability, self).get_parser(prog_name)
parser.add_argument(
'--ip-version',
type=int,
choices=[4, 6],
metavar='<ip-version>',
dest='ip_version',
help=_("List IP availability of given IP version networks"),
)
parser.add_argument(
'--project',
metavar='<project>',
help=_("List IP availability of given project"),
)
identity_common.add_project_domain_option_to_parser(parser)
return parser
def take_action(self, parsed_args):
client = self.app.client_manager.network
columns = (
'network_id',
'network_name',
'total_ips',
'used_ips',
)
column_headers = (
'Network ID',
'Network Name',
'Total IPs',
'Used IPs',
)
filters = {}
if parsed_args.ip_version:
filters['ip_version'] = parsed_args.ip_version
if parsed_args.project:
identity_client = self.app.client_manager.identity
project_id = identity_common.find_project(
identity_client,
parsed_args.project,
parsed_args.project_domain,
).id
filters['tenant_id'] = project_id
data = client.network_ip_availabilities(**filters)
return (column_headers,
(utils.get_item_properties(
s, columns,
) for s in data))
class ShowIPAvailability(command.ShowOne):
"""Show network IP availability details"""
def get_parser(self, prog_name):
parser = super(ShowIPAvailability, self).get_parser(prog_name)
parser.add_argument(
'network',
metavar="<network>",
help=_("Show IP availability for a specific network (name or ID)"),
)
return parser
def take_action(self, parsed_args):
client = self.app.client_manager.network
obj = client.find_network_ip_availability(parsed_args.network,
ignore_missing=False)
columns = _get_columns(obj)
data = utils.get_item_properties(obj, columns, formatters=_formatters)
return columns, data

View File

@ -196,6 +196,50 @@ class FakeAvailabilityZone(object):
return availability_zones return availability_zones
class FakeIPAvailability(object):
"""Fake one or more network ip availabilities."""
@staticmethod
def create_one_ip_availability():
"""Create a fake list with ip availability stats of a network.
:return:
A FakeResource object with network_name, network_id, etc.
"""
# Set default attributes.
network_ip_availability = {
'network_id': 'network-id-' + uuid.uuid4().hex,
'network_name': 'network-name-' + uuid.uuid4().hex,
'tenant_id': '',
'subnet_ip_availability': [],
'total_ips': 254,
'used_ips': 6,
}
network_ip_availability = fakes.FakeResource(
info=copy.deepcopy(network_ip_availability),
loaded=True)
return network_ip_availability
@staticmethod
def create_ip_availability(count=2):
"""Create fake list of ip availability stats of multiple networks.
:param int count:
The number of networks to fake
:return:
A list of FakeResource objects faking network ip availability stats
"""
network_ip_availabilities = []
for i in range(0, count):
network_ip_availability = \
FakeIPAvailability.create_one_ip_availability()
network_ip_availabilities.append(network_ip_availability)
return network_ip_availabilities
class FakeNetwork(object): class FakeNetwork(object):
"""Fake one or more networks.""" """Fake one or more networks."""

View File

@ -0,0 +1,180 @@
# 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 copy
import mock
from openstackclient.common import utils as osc_utils
from openstackclient.network.v2 import ip_availability
from openstackclient.tests import fakes
from openstackclient.tests.identity.v3 import fakes as identity_fakes
from openstackclient.tests.network.v2 import fakes as network_fakes
from openstackclient.tests import utils as tests_utils
class TestIPAvailability(network_fakes.TestNetworkV2):
def setUp(self):
super(TestIPAvailability, self).setUp()
# Get a shortcut to the network client
self.network = self.app.client_manager.network
# Set identity client v3. And get a shortcut to Identity client.
identity_client = identity_fakes.FakeIdentityv3Client(
endpoint=fakes.AUTH_URL,
token=fakes.AUTH_TOKEN,
)
self.app.client_manager.identity = identity_client
self.identity = self.app.client_manager.identity
# Get a shortcut to the ProjectManager Mock
self.projects_mock = self.identity.projects
self.projects_mock.get.return_value = fakes.FakeResource(
None,
copy.deepcopy(identity_fakes.PROJECT),
loaded=True,
)
class TestListIPAvailability(TestIPAvailability):
_ip_availability = \
network_fakes.FakeIPAvailability.create_ip_availability(count=3)
columns = (
'Network ID',
'Network Name',
'Total IPs',
'Used IPs',
)
data = []
for net in _ip_availability:
data.append((
net.network_id,
net.network_name,
net.total_ips,
net.used_ips,
))
def setUp(self):
super(TestListIPAvailability, self).setUp()
self.cmd = ip_availability.ListIPAvailability(
self.app, self.namespace)
self.network.network_ip_availabilities = mock.Mock(
return_value=self._ip_availability)
def test_list_no_options(self):
arglist = []
verifylist = []
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
columns, data = self.cmd.take_action(parsed_args)
self.network.network_ip_availabilities.assert_called_once_with()
self.assertEqual(self.columns, columns)
self.assertEqual(self.data, list(data))
def test_list_ip_version(self):
arglist = [
'--ip-version', str(4),
]
verifylist = [
('ip_version', 4)
]
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
columns, data = self.cmd.take_action(parsed_args)
filters = {'ip_version': 4}
self.network.network_ip_availabilities.assert_called_once_with(
**filters)
self.assertEqual(self.columns, columns)
self.assertEqual(self.data, list(data))
def test_list_project(self):
arglist = [
'--project', identity_fakes.project_name
]
verifylist = [
('project', identity_fakes.project_name)
]
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
columns, data = self.cmd.take_action(parsed_args)
filters = {'tenant_id': identity_fakes.project_id}
self.network.network_ip_availabilities.assert_called_once_with(
**filters)
self.assertEqual(self.columns, columns)
self.assertEqual(self.data, list(data))
class TestShowIPAvailability(TestIPAvailability):
_ip_availability = \
network_fakes.FakeIPAvailability.create_one_ip_availability()
columns = (
'network_id',
'network_name',
'project_id',
'subnet_ip_availability',
'total_ips',
'used_ips',
)
data = (
_ip_availability.network_id,
_ip_availability.network_name,
_ip_availability.tenant_id,
osc_utils.format_list(
_ip_availability.subnet_ip_availability),
_ip_availability.total_ips,
_ip_availability.used_ips,
)
def setUp(self):
super(TestShowIPAvailability, self).setUp()
self.network.find_network_ip_availability = mock.Mock(
return_value=self._ip_availability)
# Get the command object to test
self.cmd = ip_availability.ShowIPAvailability(
self.app, self.namespace)
def test_show_no_option(self):
arglist = []
verifylist = []
self.assertRaises(tests_utils.ParserException,
self.check_parser, self.cmd, arglist, verifylist)
def test_show_all_options(self):
arglist = [
self._ip_availability.network_name,
]
verifylist = [
('network', self._ip_availability.network_name)
]
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
columns, data = self.cmd.take_action(parsed_args)
self.network.find_network_ip_availability.assert_called_once_with(
self._ip_availability.network_name,
ignore_missing=False)
self.assertEqual(self.columns, columns)
self.assertEqual(self.data, data)

View File

@ -0,0 +1,5 @@
---
features:
- |
Add support for the ``ip availability list`` and ``ip availability show`` commands.
[Blueprint `neutron-ip-capacity <https://blueprints.launchpad.net/python-openstackclient/+spec/neutron-ip-capacity>`_]

View File

@ -333,6 +333,9 @@ openstack.network.v2 =
address_scope_set = openstackclient.network.v2.address_scope:SetAddressScope address_scope_set = openstackclient.network.v2.address_scope:SetAddressScope
address_scope_show = openstackclient.network.v2.address_scope:ShowAddressScope address_scope_show = openstackclient.network.v2.address_scope:ShowAddressScope
ip_availability_list = openstackclient.network.v2.ip_availability:ListIPAvailability
ip_availability_show = openstackclient.network.v2.ip_availability:ShowIPAvailability
ip_floating_create = openstackclient.network.v2.floating_ip:CreateFloatingIP ip_floating_create = openstackclient.network.v2.floating_ip:CreateFloatingIP
ip_floating_delete = openstackclient.network.v2.floating_ip:DeleteFloatingIP ip_floating_delete = openstackclient.network.v2.floating_ip:DeleteFloatingIP
ip_floating_list = openstackclient.network.v2.floating_ip:ListFloatingIP ip_floating_list = openstackclient.network.v2.floating_ip:ListFloatingIP