Merge "Fix ipv6 interface configuration"
This commit is contained in:
commit
1bc108b19f
@ -22,6 +22,8 @@ import time
|
|||||||
from oslo_config import cfg
|
from oslo_config import cfg
|
||||||
from oslo_log import log as logging
|
from oslo_log import log as logging
|
||||||
import pyroute2
|
import pyroute2
|
||||||
|
# pylint: disable=no-name-in-module
|
||||||
|
from pyroute2.netlink.rtnl import ifaddrmsg
|
||||||
|
|
||||||
from octavia.amphorae.backends.utils import interface_file
|
from octavia.amphorae.backends.utils import interface_file
|
||||||
from octavia.common import constants as consts
|
from octavia.common import constants as consts
|
||||||
@ -37,6 +39,9 @@ class InterfaceController(object):
|
|||||||
DELETE = 'delete'
|
DELETE = 'delete'
|
||||||
SET = 'set'
|
SET = 'set'
|
||||||
|
|
||||||
|
TENTATIVE_WAIT_INTERVAL = .2
|
||||||
|
TENTATIVE_WAIT_TIMEOUT = 30
|
||||||
|
|
||||||
def interface_file_list(self):
|
def interface_file_list(self):
|
||||||
net_dir = interface_file.InterfaceFile.get_directory()
|
net_dir = interface_file.InterfaceFile.get_directory()
|
||||||
|
|
||||||
@ -134,6 +139,21 @@ class InterfaceController(object):
|
|||||||
LOG.debug("Running '%s'", cmd)
|
LOG.debug("Running '%s'", cmd)
|
||||||
subprocess.check_output(cmd, stderr=subprocess.STDOUT)
|
subprocess.check_output(cmd, stderr=subprocess.STDOUT)
|
||||||
|
|
||||||
|
def _wait_tentative(self, ipr, idx):
|
||||||
|
start = time.time()
|
||||||
|
while time.time() - start < self.TENTATIVE_WAIT_TIMEOUT:
|
||||||
|
addrs = ipr.get_addr(idx)
|
||||||
|
has_tentative = [
|
||||||
|
True
|
||||||
|
for addr in addrs
|
||||||
|
if (addr['family'] == socket.AF_INET6 and
|
||||||
|
addr['flags'] & ifaddrmsg.IFA_F_TENTATIVE)]
|
||||||
|
if not has_tentative:
|
||||||
|
return
|
||||||
|
time.sleep(self.TENTATIVE_WAIT_INTERVAL)
|
||||||
|
LOG.warning("Some IPV6 addresses remain still in 'tentative' state "
|
||||||
|
"after %d seconds.", self.TENTATIVE_WAIT_TIMEOUT)
|
||||||
|
|
||||||
def up(self, interface):
|
def up(self, interface):
|
||||||
LOG.info("Setting interface %s up", interface.name)
|
LOG.info("Setting interface %s up", interface.name)
|
||||||
|
|
||||||
@ -158,6 +178,8 @@ class InterfaceController(object):
|
|||||||
LOG.debug("%s: Adding address %s", interface.name, address)
|
LOG.debug("%s: Adding address %s", interface.name, address)
|
||||||
self._ipr_command(ipr.addr, self.ADD, index=idx, **address)
|
self._ipr_command(ipr.addr, self.ADD, index=idx, **address)
|
||||||
|
|
||||||
|
self._wait_tentative(ipr, idx)
|
||||||
|
|
||||||
for route in interface.routes:
|
for route in interface.routes:
|
||||||
route[consts.FAMILY] = self._family(route[consts.DST])
|
route[consts.FAMILY] = self._family(route[consts.DST])
|
||||||
LOG.debug("%s: Adding route %s", interface.name, route)
|
LOG.debug("%s: Adding route %s", interface.name, route)
|
||||||
|
@ -446,8 +446,11 @@ class TestInterface(base.TestCase):
|
|||||||
@mock.patch('pyroute2.IPRoute.link')
|
@mock.patch('pyroute2.IPRoute.link')
|
||||||
@mock.patch('pyroute2.IPRoute.link_lookup')
|
@mock.patch('pyroute2.IPRoute.link_lookup')
|
||||||
@mock.patch('subprocess.check_output')
|
@mock.patch('subprocess.check_output')
|
||||||
def test_up_auto(self, mock_check_output, mock_link_lookup, mock_link,
|
@mock.patch('octavia.amphorae.backends.utils.interface.'
|
||||||
mock_addr, mock_route, mock_rule):
|
'InterfaceController._wait_tentative')
|
||||||
|
def test_up_auto(self, mock_wait_tentative, mock_check_output,
|
||||||
|
mock_link_lookup, mock_link, mock_addr, mock_route,
|
||||||
|
mock_rule):
|
||||||
iface = interface_file.InterfaceFile(
|
iface = interface_file.InterfaceFile(
|
||||||
name="eth1",
|
name="eth1",
|
||||||
mtu=1450,
|
mtu=1450,
|
||||||
@ -900,3 +903,50 @@ class TestInterface(base.TestCase):
|
|||||||
stderr=-2),
|
stderr=-2),
|
||||||
mock.call(["post-down", iface.name])
|
mock.call(["post-down", iface.name])
|
||||||
])
|
])
|
||||||
|
|
||||||
|
@mock.patch("time.time")
|
||||||
|
@mock.patch("time.sleep")
|
||||||
|
def test__wait_tentative(self, mock_time_sleep, mock_time_time):
|
||||||
|
mock_ipr = mock.MagicMock()
|
||||||
|
mock_ipr.get_addr.side_effect = [
|
||||||
|
({'family': socket.AF_INET,
|
||||||
|
'flags': 0},
|
||||||
|
{'family': socket.AF_INET6,
|
||||||
|
'flags': 0x40}, # tentative
|
||||||
|
{'family': socket.AF_INET6,
|
||||||
|
'flags': 0}),
|
||||||
|
({'family': socket.AF_INET,
|
||||||
|
'flags': 0},
|
||||||
|
{'family': socket.AF_INET6,
|
||||||
|
'flags': 0},
|
||||||
|
{'family': socket.AF_INET6,
|
||||||
|
'flags': 0})
|
||||||
|
]
|
||||||
|
|
||||||
|
mock_time_time.return_value = 0
|
||||||
|
|
||||||
|
controller = interface.InterfaceController()
|
||||||
|
idx = 4
|
||||||
|
|
||||||
|
controller._wait_tentative(mock_ipr, idx)
|
||||||
|
mock_time_sleep.assert_called_once()
|
||||||
|
|
||||||
|
@mock.patch("time.time")
|
||||||
|
@mock.patch("time.sleep")
|
||||||
|
def test__wait_tentative_timeout(self, mock_time_sleep,
|
||||||
|
mock_time_time):
|
||||||
|
mock_ipr = mock.MagicMock()
|
||||||
|
mock_ipr.get_addr.return_value = (
|
||||||
|
{'family': socket.AF_INET6,
|
||||||
|
'flags': 0x40}, # tentative
|
||||||
|
{'family': socket.AF_INET6,
|
||||||
|
'flags': 0}
|
||||||
|
)
|
||||||
|
|
||||||
|
mock_time_time.side_effect = [0, 0, 1, 2, 29, 30, 31]
|
||||||
|
|
||||||
|
controller = interface.InterfaceController()
|
||||||
|
idx = 4
|
||||||
|
|
||||||
|
controller._wait_tentative(mock_ipr, idx)
|
||||||
|
self.assertEqual(4, len(mock_time_sleep.mock_calls))
|
||||||
|
@ -0,0 +1,5 @@
|
|||||||
|
---
|
||||||
|
fixes:
|
||||||
|
- |
|
||||||
|
Fix an issue with IPv6 members that could have been set in operating_status
|
||||||
|
``ERROR`` just after being added.
|
Loading…
x
Reference in New Issue
Block a user