Introduce overwrite functionality in `osc port set`

The overwrite functionality allows user to overwrite
either the binding-profile or the fixed-ips of a
specific port.

Change-Id: I8ec3d04eeaf28972ee545fcdda4d5f7bd9deb915
partially-implements: blueprint allow-overwrite-set-options
This commit is contained in:
reedip 2016-04-20 17:31:52 +09:00 committed by Reedip
parent 320ed01774
commit b38be94a5d
4 changed files with 78 additions and 15 deletions

View File

@ -135,11 +135,13 @@ Set port properties
.. code:: bash .. code:: bash
os port set os port set
[--fixed-ip subnet=<subnet>,ip-address=<ip-address> | --no-fixed-ip] [--fixed-ip subnet=<subnet>,ip-address=<ip-address>]
[--no-fixed-ip]
[--device <device-id>] [--device <device-id>]
[--device-owner <device-owner>] [--device-owner <device-owner>]
[--vnic-type <vnic-type>] [--vnic-type <vnic-type>]
[--binding-profile <binding-profile> | --no-binding-profile] [--binding-profile <binding-profile>]
[--no-binding-profile]
[--host <host-id>] [--host <host-id>]
[--enable | --disable] [--enable | --disable]
[--name <name>] [--name <name>]
@ -153,7 +155,9 @@ Set port properties
.. option:: --no-fixed-ip .. option:: --no-fixed-ip
Clear existing information of fixed IP addresses Clear existing information of fixed IP addresses.
Specify both --fixed-ip and --no-fixed-ip
to overwrite the current fixed IP addresses.
.. option:: --device <device-id> .. option:: --device <device-id>
@ -177,7 +181,9 @@ Set port properties
.. option:: --no-binding-profile .. option:: --no-binding-profile
Clear existing information of binding:profile Clear existing information of binding:profile.
Specify both --binding-profile and --no-binding-profile
to overwrite the current binding:profile information.
.. option:: --host <host-id> .. option:: --host <host-id>

View File

@ -109,7 +109,6 @@ def _get_attrs(client_manager, parsed_args):
'The --host-id option is deprecated, ' 'The --host-id option is deprecated, '
'please use --host instead.' 'please use --host instead.'
)) ))
if parsed_args.fixed_ip is not None: if parsed_args.fixed_ip is not None:
attrs['fixed_ips'] = parsed_args.fixed_ip attrs['fixed_ips'] = parsed_args.fixed_ip
if parsed_args.device: if parsed_args.device:
@ -409,8 +408,7 @@ class SetPort(command.Command):
metavar="<name>", metavar="<name>",
help=_("Set port name") help=_("Set port name")
) )
fixed_ip = parser.add_mutually_exclusive_group() parser.add_argument(
fixed_ip.add_argument(
'--fixed-ip', '--fixed-ip',
metavar='subnet=<subnet>,ip-address=<ip-address>', metavar='subnet=<subnet>,ip-address=<ip-address>',
action=parseractions.MultiKeyValueAction, action=parseractions.MultiKeyValueAction,
@ -419,13 +417,14 @@ class SetPort(command.Command):
"subnet=<subnet>,ip-address=<ip-address> " "subnet=<subnet>,ip-address=<ip-address> "
"(repeat option to set multiple fixed IP addresses)") "(repeat option to set multiple fixed IP addresses)")
) )
fixed_ip.add_argument( parser.add_argument(
'--no-fixed-ip', '--no-fixed-ip',
action='store_true', action='store_true',
help=_("Clear existing information of fixed IP addresses") help=_("Clear existing information of fixed IP addresses."
"Specify both --fixed-ip and --no-fixed-ip "
"to overwrite the current fixed IP addresses.")
) )
binding_profile = parser.add_mutually_exclusive_group() parser.add_argument(
binding_profile.add_argument(
'--binding-profile', '--binding-profile',
metavar='<binding-profile>', metavar='<binding-profile>',
action=JSONKeyValueAction, action=JSONKeyValueAction,
@ -433,10 +432,12 @@ class SetPort(command.Command):
"be passed as <key>=<value> or JSON. " "be passed as <key>=<value> or JSON. "
"(repeat option to set multiple binding:profile data)") "(repeat option to set multiple binding:profile data)")
) )
binding_profile.add_argument( parser.add_argument(
'--no-binding-profile', '--no-binding-profile',
action='store_true', action='store_true',
help=_("Clear existing information of binding:profile") help=_("Clear existing information of binding:profile."
"Specify both --binding-profile and --no-binding-profile "
"to overwrite the current binding:profile information.")
) )
parser.add_argument( parser.add_argument(
'port', 'port',
@ -452,7 +453,11 @@ class SetPort(command.Command):
attrs = _get_attrs(self.app.client_manager, parsed_args) attrs = _get_attrs(self.app.client_manager, parsed_args)
obj = client.find_port(parsed_args.port, ignore_missing=False) obj = client.find_port(parsed_args.port, ignore_missing=False)
if 'binding:profile' in attrs: if 'binding:profile' in attrs:
attrs['binding:profile'].update(obj.binding_profile) # Do not modify attrs if both binding_profile/no_binding given
if not parsed_args.no_binding_profile:
tmp_binding_profile = copy.deepcopy(obj.binding_profile)
tmp_binding_profile.update(attrs['binding:profile'])
attrs['binding:profile'] = tmp_binding_profile
elif parsed_args.no_binding_profile: elif parsed_args.no_binding_profile:
attrs['binding:profile'] = {} attrs['binding:profile'] = {}
if 'fixed_ips' in attrs: if 'fixed_ips' in attrs:
@ -461,7 +466,9 @@ class SetPort(command.Command):
# would therefore add an empty dictionary, while we need # would therefore add an empty dictionary, while we need
# to append the attrs['fixed_ips'] iff there is some info # to append the attrs['fixed_ips'] iff there is some info
# in the obj.fixed_ips. Therefore I have opted for this `for` loop # in the obj.fixed_ips. Therefore I have opted for this `for` loop
attrs['fixed_ips'] += [ip for ip in obj.fixed_ips if ip] # Do not modify attrs if fixed_ip/no_fixed_ip given
if not parsed_args.no_fixed_ip:
attrs['fixed_ips'] += [ip for ip in obj.fixed_ips if ip]
elif parsed_args.no_fixed_ip: elif parsed_args.no_fixed_ip:
attrs['fixed_ips'] = [] attrs['fixed_ips'] = []

View File

@ -465,6 +465,50 @@ class TestSetPort(TestPort):
self.network.update_port.assert_called_once_with(_testport, **attrs) self.network.update_port.assert_called_once_with(_testport, **attrs)
self.assertIsNone(result) self.assertIsNone(result)
def test_overwrite_binding_profile(self):
_testport = network_fakes.FakePort.create_one_port(
{'binding_profile': {'lok_i': 'visi_on'}})
self.network.find_port = mock.Mock(return_value=_testport)
arglist = [
'--binding-profile', 'lok_i=than_os',
'--no-binding-profile',
_testport.name,
]
verifylist = [
('binding_profile', {'lok_i': 'than_os'}),
('no_binding_profile', True)
]
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
result = self.cmd.take_action(parsed_args)
attrs = {
'binding:profile':
{'lok_i': 'than_os'},
}
self.network.update_port.assert_called_once_with(_testport, **attrs)
self.assertIsNone(result)
def test_overwrite_fixed_ip(self):
_testport = network_fakes.FakePort.create_one_port(
{'fixed_ips': [{'ip_address': '0.0.0.1'}]})
self.network.find_port = mock.Mock(return_value=_testport)
arglist = [
'--fixed-ip', 'ip-address=10.0.0.12',
'--no-fixed-ip',
_testport.name,
]
verifylist = [
('fixed_ip', [{'ip-address': '10.0.0.12'}]),
('no_fixed_ip', True)
]
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
result = self.cmd.take_action(parsed_args)
attrs = {
'fixed_ips': [
{'ip_address': '10.0.0.12'}],
}
self.network.update_port.assert_called_once_with(_testport, **attrs)
self.assertIsNone(result)
def test_set_this(self): def test_set_this(self):
arglist = [ arglist = [
'--disable', '--disable',

View File

@ -0,0 +1,6 @@
---
features:
- |
``port set`` command now allows the user to overwrite fixed-ips or binding-profile
of a port.
[ Blueprint `allow-overwrite-set-options <https://blueprints.launchpad.net/python-openstackclient/+spec/allow-overwrite-set-options>` _]