Refactor ovn_bgp_agent.privileged.ovs_vsctl
When ovs commands fail, we re-try after adding `-O OpenFlow13`, but that is only valid for `ovs-ofctl`. To fix this issue, this patch implements a small refactor of the module in order to separete calls to ovs-ofctl and ovs-vsctl. Related-Bug: #2080258 Change-Id: Ib8ee872d4bd587ade325bc4f6da3e9d8804c28c0
This commit is contained in:
parent
e615106ab8
commit
7e348daca0
@ -34,8 +34,8 @@ def _find_ovs_port(bridge):
|
|||||||
# TODO(ltomasbo): What happens if there are several patch ports on the
|
# TODO(ltomasbo): What happens if there are several patch ports on the
|
||||||
# same bridge?
|
# same bridge?
|
||||||
ovs_port = None
|
ovs_port = None
|
||||||
ovs_ports = ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd(
|
ovs_ports = ovn_bgp_agent.privileged.ovs_vsctl.ovs_vsctl(
|
||||||
'ovs-vsctl', ['list-ports', bridge])[0].rstrip()
|
['list-ports', bridge])[0].rstrip()
|
||||||
for p in ovs_ports.split('\n'):
|
for p in ovs_ports.split('\n'):
|
||||||
if p.startswith(constants.OVS_PATCH_PROVNET_PORT_PREFIX):
|
if p.startswith(constants.OVS_PATCH_PROVNET_PORT_PREFIX):
|
||||||
ovs_port = p
|
ovs_port = p
|
||||||
@ -46,8 +46,8 @@ def get_bridge_flows(bridge, filter_=None):
|
|||||||
args = ['dump-flows', bridge]
|
args = ['dump-flows', bridge]
|
||||||
if filter_ is not None:
|
if filter_ is not None:
|
||||||
args.append(filter_)
|
args.append(filter_)
|
||||||
return ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd(
|
return ovn_bgp_agent.privileged.ovs_vsctl.ovs_ofctl(
|
||||||
'ovs-ofctl', args)[0].split('\n')[1:-1]
|
args)[0].split('\n')[1:-1]
|
||||||
|
|
||||||
|
|
||||||
@tenacity.retry(
|
@tenacity.retry(
|
||||||
@ -57,8 +57,8 @@ def get_bridge_flows(bridge, filter_=None):
|
|||||||
reraise=True)
|
reraise=True)
|
||||||
def get_device_port_at_ovs(device):
|
def get_device_port_at_ovs(device):
|
||||||
try:
|
try:
|
||||||
ofport = ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd(
|
ofport = ovn_bgp_agent.privileged.ovs_vsctl.ovs_vsctl(
|
||||||
'ovs-vsctl', ['get', 'Interface', device, 'ofport']
|
['get', 'Interface', device, 'ofport']
|
||||||
)[0].rstrip()
|
)[0].rstrip()
|
||||||
except Exception:
|
except Exception:
|
||||||
raise agent_exc.PortNotFound(port=device)
|
raise agent_exc.PortNotFound(port=device)
|
||||||
@ -71,8 +71,8 @@ def get_device_port_at_ovs(device):
|
|||||||
|
|
||||||
|
|
||||||
def get_ovs_ports_info(bridge):
|
def get_ovs_ports_info(bridge):
|
||||||
ovs_ports = ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd(
|
ovs_ports = ovn_bgp_agent.privileged.ovs_vsctl.ovs_vsctl(
|
||||||
'ovs-vsctl', ['list-ports', bridge])[0].rstrip()
|
['list-ports', bridge])[0].rstrip()
|
||||||
return ovs_ports.split("\n")
|
return ovs_ports.split("\n")
|
||||||
|
|
||||||
|
|
||||||
@ -113,11 +113,11 @@ def ensure_mac_tweak_flows(bridge, mac, ports, cookie):
|
|||||||
exist_flow_v6 = True
|
exist_flow_v6 = True
|
||||||
|
|
||||||
if not exist_flow:
|
if not exist_flow:
|
||||||
ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd(
|
ovn_bgp_agent.privileged.ovs_vsctl.ovs_ofctl(
|
||||||
'ovs-ofctl', ['add-flow', bridge, flow])
|
['add-flow', bridge, flow])
|
||||||
if not exist_flow_v6:
|
if not exist_flow_v6:
|
||||||
ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd(
|
ovn_bgp_agent.privileged.ovs_vsctl.ovs_ofctl(
|
||||||
'ovs-ofctl', ['add-flow', bridge, flow_v6])
|
['add-flow', bridge, flow_v6])
|
||||||
|
|
||||||
|
|
||||||
def remove_extra_ovs_flows(ovs_flows, bridge, cookie):
|
def remove_extra_ovs_flows(ovs_flows, bridge, cookie):
|
||||||
@ -143,13 +143,12 @@ def remove_extra_ovs_flows(ovs_flows, bridge, cookie):
|
|||||||
if flow.split("priority")[1] not in expected_flows:
|
if flow.split("priority")[1] not in expected_flows:
|
||||||
del_flow = ('{},{}').format(
|
del_flow = ('{},{}').format(
|
||||||
cookie_id, flow.split("priority=900,")[1].split(" actions")[0])
|
cookie_id, flow.split("priority=900,")[1].split(" actions")[0])
|
||||||
ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd(
|
ovn_bgp_agent.privileged.ovs_vsctl.ovs_ofctl(
|
||||||
'ovs-ofctl', ['del-flows', bridge, del_flow])
|
['del-flows', bridge, del_flow])
|
||||||
|
|
||||||
|
|
||||||
def ensure_flow(bridge, flow):
|
def ensure_flow(bridge, flow):
|
||||||
ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd(
|
ovn_bgp_agent.privileged.ovs_vsctl.ovs_ofctl(['add-flow', bridge, flow])
|
||||||
'ovs-ofctl', ['add-flow', bridge, flow])
|
|
||||||
|
|
||||||
|
|
||||||
def ensure_evpn_ovs_flow(bridge, cookie, mac, output_port, port_dst, net,
|
def ensure_evpn_ovs_flow(bridge, cookie, mac, output_port, port_dst, net,
|
||||||
@ -175,8 +174,7 @@ def ensure_evpn_ovs_flow(bridge, cookie, mac, output_port, port_dst, net,
|
|||||||
"actions=mod_dl_dst:{},{}output={}".format(
|
"actions=mod_dl_dst:{},{}output={}".format(
|
||||||
cookie, ovs_ofport, mac, net, port_dst_mac, strip_vlan_opt,
|
cookie, ovs_ofport, mac, net, port_dst_mac, strip_vlan_opt,
|
||||||
vrf_ofport))
|
vrf_ofport))
|
||||||
ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd(
|
ovn_bgp_agent.privileged.ovs_vsctl.ovs_ofctl(['add-flow', bridge, flow])
|
||||||
'ovs-ofctl', ['add-flow', bridge, flow])
|
|
||||||
|
|
||||||
|
|
||||||
def remove_evpn_router_ovs_flows(bridge, cookie, mac):
|
def remove_evpn_router_ovs_flows(bridge, cookie, mac):
|
||||||
@ -187,13 +185,12 @@ def remove_evpn_router_ovs_flows(bridge, cookie, mac):
|
|||||||
cookie_id = "cookie={}/-1".format(cookie)
|
cookie_id = "cookie={}/-1".format(cookie)
|
||||||
flow = ("{},ip,in_port={},dl_src:{}".format(
|
flow = ("{},ip,in_port={},dl_src:{}".format(
|
||||||
cookie_id, ovs_ofport, mac))
|
cookie_id, ovs_ofport, mac))
|
||||||
ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd(
|
ovn_bgp_agent.privileged.ovs_vsctl.ovs_ofctl(['del-flows', bridge, flow])
|
||||||
'ovs-ofctl', ['del-flows', bridge, flow])
|
|
||||||
|
|
||||||
flow_v6 = ("{},ipv6,in_port={},dl_src:{}".format(cookie_id, ovs_ofport,
|
flow_v6 = ("{},ipv6,in_port={},dl_src:{}".format(cookie_id, ovs_ofport,
|
||||||
mac))
|
mac))
|
||||||
ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd(
|
ovn_bgp_agent.privileged.ovs_vsctl.ovs_ofctl(
|
||||||
'ovs-ofctl', ['del-flows', bridge, flow_v6])
|
['del-flows', bridge, flow_v6])
|
||||||
|
|
||||||
|
|
||||||
def remove_evpn_network_ovs_flow(bridge, cookie, mac, net):
|
def remove_evpn_network_ovs_flow(bridge, cookie, mac, net):
|
||||||
@ -209,15 +206,14 @@ def remove_evpn_network_ovs_flow(bridge, cookie, mac, net):
|
|||||||
else:
|
else:
|
||||||
flow = ("{},ip,in_port={},dl_src:{},nw_src={}".format(
|
flow = ("{},ip,in_port={},dl_src:{},nw_src={}".format(
|
||||||
cookie_id, ovs_ofport, mac, net))
|
cookie_id, ovs_ofport, mac, net))
|
||||||
ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd(
|
ovn_bgp_agent.privileged.ovs_vsctl.ovs_ofctl(['del-flows', bridge, flow])
|
||||||
'ovs-ofctl', ['del-flows', bridge, flow])
|
|
||||||
|
|
||||||
|
|
||||||
def add_device_to_ovs_bridge(device, bridge, vlan_tag=None):
|
def add_device_to_ovs_bridge(device, bridge, vlan_tag=None):
|
||||||
args = ['--may-exist', 'add-port', bridge, device]
|
args = ['--may-exist', 'add-port', bridge, device]
|
||||||
if vlan_tag is not None:
|
if vlan_tag is not None:
|
||||||
args.append('tag=%s' % vlan_tag)
|
args.append('tag=%s' % vlan_tag)
|
||||||
ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd('ovs-vsctl', args)
|
ovn_bgp_agent.privileged.ovs_vsctl.ovs_vsctl(args)
|
||||||
|
|
||||||
|
|
||||||
def del_device_from_ovs_bridge(device, bridge=None):
|
def del_device_from_ovs_bridge(device, bridge=None):
|
||||||
@ -225,7 +221,7 @@ def del_device_from_ovs_bridge(device, bridge=None):
|
|||||||
if bridge:
|
if bridge:
|
||||||
args.append(bridge)
|
args.append(bridge)
|
||||||
args.append(device)
|
args.append(device)
|
||||||
ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd('ovs-vsctl', args)
|
ovn_bgp_agent.privileged.ovs_vsctl.ovs_vsctl(args)
|
||||||
|
|
||||||
|
|
||||||
def add_vlan_port_to_ovs_bridge(bridge, vlan, vlan_tag):
|
def add_vlan_port_to_ovs_bridge(bridge, vlan, vlan_tag):
|
||||||
@ -234,15 +230,15 @@ def add_vlan_port_to_ovs_bridge(bridge, vlan, vlan_tag):
|
|||||||
args = [
|
args = [
|
||||||
'--may-exist', 'add-port', bridge, vlan, 'tag={}'.format(vlan_tag),
|
'--may-exist', 'add-port', bridge, vlan, 'tag={}'.format(vlan_tag),
|
||||||
'--', 'set', 'interface', vlan, 'type=internal']
|
'--', 'set', 'interface', vlan, 'type=internal']
|
||||||
ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd('ovs-vsctl', args)
|
ovn_bgp_agent.privileged.ovs_vsctl.ovs_vsctl(args)
|
||||||
|
|
||||||
|
|
||||||
def del_flow(flow, bridge, cookie):
|
def del_flow(flow, bridge, cookie):
|
||||||
cookie_id = "cookie={}/-1".format(cookie)
|
cookie_id = "cookie={}/-1".format(cookie)
|
||||||
f = '{},priority{}'.format(
|
f = '{},priority{}'.format(
|
||||||
cookie_id, flow.split(' actions')[0].split(' priority')[1])
|
cookie_id, flow.split(' actions')[0].split(' priority')[1])
|
||||||
ovn_bgp_agent.privileged.ovs_vsctl.ovs_cmd(
|
ovn_bgp_agent.privileged.ovs_vsctl.ovs_ofctl(
|
||||||
'ovs-ofctl', ['--strict', 'del-flows', bridge, f])
|
['--strict', 'del-flows', bridge, f])
|
||||||
|
|
||||||
|
|
||||||
def get_flow_info(flow):
|
def get_flow_info(flow):
|
||||||
|
@ -28,15 +28,17 @@ def ovs_cmd(command, args, timeout=None):
|
|||||||
full_args += args
|
full_args += args
|
||||||
try:
|
try:
|
||||||
return processutils.execute(*full_args)
|
return processutils.execute(*full_args)
|
||||||
except processutils.ProcessExecutionError:
|
except Exception:
|
||||||
full_args += ['-O', 'OpenFlow13']
|
LOG.error("Unable to execute %s %s", command, full_args)
|
||||||
|
raise
|
||||||
|
|
||||||
|
|
||||||
|
def ovs_vsctl(args, timeout=None):
|
||||||
|
return ovs_cmd('ovs-vsctl', args, timeout)
|
||||||
|
|
||||||
|
|
||||||
|
def ovs_ofctl(args, timeout=None):
|
||||||
try:
|
try:
|
||||||
return processutils.execute(*full_args)
|
return ovs_cmd('ovs-ofctl', args, timeout)
|
||||||
except Exception as e:
|
except processutils.ProcessExecutionError:
|
||||||
LOG.exception("Unable to execute %s %s. Exception: %s",
|
return ovs_cmd('ovs-ofctl', args + ['-O', 'OpenFlow13'], timeout)
|
||||||
command, full_args, e)
|
|
||||||
raise
|
|
||||||
except Exception as e:
|
|
||||||
LOG.exception("Unable to execute %s %s. Exception: %s", command,
|
|
||||||
full_args, e)
|
|
||||||
raise
|
|
||||||
|
@ -44,7 +44,7 @@ class TestOVS(test_base.TestCase):
|
|||||||
fake_flow_1 = '{},ipv6,in_port={}'.format(self.cookie_id, port_iface)
|
fake_flow_1 = '{},ipv6,in_port={}'.format(self.cookie_id, port_iface)
|
||||||
fake_filter = 'cookie=fake-cookie/-1' if has_filter else None
|
fake_filter = 'cookie=fake-cookie/-1' if has_filter else None
|
||||||
flows = 'HEADER\n%s\n%s\n' % (fake_flow_0, fake_flow_1)
|
flows = 'HEADER\n%s\n%s\n' % (fake_flow_0, fake_flow_1)
|
||||||
self.mock_ovs_vsctl.ovs_cmd.return_value = [flows]
|
self.mock_ovs_vsctl.ovs_ofctl.return_value = [flows]
|
||||||
|
|
||||||
ret = ovs_utils.get_bridge_flows(self.bridge, filter_=fake_filter)
|
ret = ovs_utils.get_bridge_flows(self.bridge, filter_=fake_filter)
|
||||||
|
|
||||||
@ -52,8 +52,7 @@ class TestOVS(test_base.TestCase):
|
|||||||
if has_filter:
|
if has_filter:
|
||||||
expected_args.append(fake_filter)
|
expected_args.append(fake_filter)
|
||||||
|
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_called_once_with(
|
self.mock_ovs_vsctl.ovs_ofctl.assert_called_once_with(expected_args)
|
||||||
'ovs-ofctl', expected_args)
|
|
||||||
self.assertEqual([fake_flow_0, fake_flow_1], ret)
|
self.assertEqual([fake_flow_0, fake_flow_1], ret)
|
||||||
|
|
||||||
def test_get_bridge_flows(self):
|
def test_get_bridge_flows(self):
|
||||||
@ -65,77 +64,76 @@ class TestOVS(test_base.TestCase):
|
|||||||
def test_get_device_port_at_ovs(self):
|
def test_get_device_port_at_ovs(self):
|
||||||
port = 'fake-port'
|
port = 'fake-port'
|
||||||
port_iface = '1'
|
port_iface = '1'
|
||||||
self.mock_ovs_vsctl.ovs_cmd.return_value = port_iface
|
self.mock_ovs_vsctl.ovs_vsctl.return_value = port_iface
|
||||||
|
|
||||||
ret = ovs_utils.get_device_port_at_ovs(port)
|
ret = ovs_utils.get_device_port_at_ovs(port)
|
||||||
|
|
||||||
self.assertEqual(port_iface, ret)
|
self.assertEqual(port_iface, ret)
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_called_once_with(
|
self.mock_ovs_vsctl.ovs_vsctl.assert_called_once_with(
|
||||||
'ovs-vsctl', ['get', 'Interface', port, 'ofport'])
|
['get', 'Interface', port, 'ofport'])
|
||||||
|
|
||||||
def test_get_ovs_ports_info(self):
|
def test_get_ovs_ports_info(self):
|
||||||
bridge = 'fake-bridge'
|
bridge = 'fake-bridge'
|
||||||
bridge_ports = ['br-ex']
|
bridge_ports = ['br-ex']
|
||||||
self.mock_ovs_vsctl.ovs_cmd.return_value = bridge_ports
|
self.mock_ovs_vsctl.ovs_vsctl.return_value = bridge_ports
|
||||||
|
|
||||||
ret = ovs_utils.get_ovs_ports_info(bridge)
|
ret = ovs_utils.get_ovs_ports_info(bridge)
|
||||||
|
|
||||||
self.assertEqual(bridge_ports, ret)
|
self.assertEqual(bridge_ports, ret)
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_called_once_with(
|
self.mock_ovs_vsctl.ovs_vsctl.assert_called_once_with(
|
||||||
'ovs-vsctl', ['list-ports', bridge])
|
['list-ports', bridge])
|
||||||
|
|
||||||
def test_get_ovs_patch_port_ofport(self):
|
def test_get_ovs_patch_port_ofport(self):
|
||||||
patch = 'fake-patch'
|
patch = 'fake-patch'
|
||||||
ofport = ['1']
|
ofport = ['1']
|
||||||
self.mock_ovs_vsctl.ovs_cmd.return_value = ofport
|
self.mock_ovs_vsctl.ovs_vsctl.return_value = ofport
|
||||||
|
|
||||||
ret = ovs_utils.get_ovs_patch_port_ofport(patch)
|
ret = ovs_utils.get_ovs_patch_port_ofport(patch)
|
||||||
|
|
||||||
self.assertEqual(ofport[0], ret)
|
self.assertEqual(ofport[0], ret)
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_called_once_with(
|
self.mock_ovs_vsctl.ovs_vsctl.assert_called_once_with(
|
||||||
'ovs-vsctl',
|
|
||||||
['get', 'Interface', 'patch-fake-patch-to-br-int', 'ofport'])
|
['get', 'Interface', 'patch-fake-patch-to-br-int', 'ofport'])
|
||||||
|
|
||||||
def test_get_ovs_patch_port_ofport_exception(self):
|
def test_get_ovs_patch_port_ofport_exception(self):
|
||||||
patch = 'fake-patch'
|
patch = 'fake-patch'
|
||||||
self.mock_ovs_vsctl.ovs_cmd.side_effect = Exception
|
self.mock_ovs_vsctl.ovs_vsctl.side_effect = Exception
|
||||||
|
|
||||||
self.assertRaises(agent_exc.PortNotFound,
|
self.assertRaises(agent_exc.PortNotFound,
|
||||||
ovs_utils.get_ovs_patch_port_ofport, patch)
|
ovs_utils.get_ovs_patch_port_ofport, patch)
|
||||||
|
|
||||||
expected_calls = [
|
expected_calls = [
|
||||||
mock.call('ovs-vsctl', ['get', 'Interface',
|
mock.call(['get', 'Interface',
|
||||||
'patch-fake-patch-to-br-int', 'ofport']),
|
'patch-fake-patch-to-br-int', 'ofport']),
|
||||||
mock.call('ovs-vsctl', ['get', 'Interface',
|
mock.call(['get', 'Interface',
|
||||||
'patch-fake-patch-to-br-int', 'ofport']),
|
'patch-fake-patch-to-br-int', 'ofport']),
|
||||||
mock.call('ovs-vsctl', ['get', 'Interface',
|
mock.call(['get', 'Interface',
|
||||||
'patch-fake-patch-to-br-int', 'ofport']),
|
'patch-fake-patch-to-br-int', 'ofport']),
|
||||||
mock.call('ovs-vsctl', ['get', 'Interface',
|
mock.call(['get', 'Interface',
|
||||||
'patch-fake-patch-to-br-int', 'ofport']),
|
'patch-fake-patch-to-br-int', 'ofport']),
|
||||||
mock.call('ovs-vsctl', ['get', 'Interface',
|
mock.call(['get', 'Interface',
|
||||||
'patch-fake-patch-to-br-int', 'ofport'])]
|
'patch-fake-patch-to-br-int', 'ofport'])]
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_has_calls(expected_calls)
|
self.mock_ovs_vsctl.ovs_vsctl.assert_has_calls(expected_calls)
|
||||||
|
|
||||||
def test_get_ovs_patch_port_ofport_no_port(self):
|
def test_get_ovs_patch_port_ofport_no_port(self):
|
||||||
patch = 'fake-patch'
|
patch = 'fake-patch'
|
||||||
ofport = ['[]']
|
ofport = ['[]']
|
||||||
self.mock_ovs_vsctl.ovs_cmd.return_value = ofport
|
self.mock_ovs_vsctl.ovs_vsctl.return_value = ofport
|
||||||
|
|
||||||
self.assertRaises(agent_exc.PortNotFound,
|
self.assertRaises(agent_exc.PortNotFound,
|
||||||
ovs_utils.get_ovs_patch_port_ofport, patch)
|
ovs_utils.get_ovs_patch_port_ofport, patch)
|
||||||
|
|
||||||
expected_calls = [
|
expected_calls = [
|
||||||
mock.call('ovs-vsctl', ['get', 'Interface',
|
mock.call(['get', 'Interface',
|
||||||
'patch-fake-patch-to-br-int', 'ofport']),
|
'patch-fake-patch-to-br-int', 'ofport']),
|
||||||
mock.call('ovs-vsctl', ['get', 'Interface',
|
mock.call(['get', 'Interface',
|
||||||
'patch-fake-patch-to-br-int', 'ofport']),
|
'patch-fake-patch-to-br-int', 'ofport']),
|
||||||
mock.call('ovs-vsctl', ['get', 'Interface',
|
mock.call(['get', 'Interface',
|
||||||
'patch-fake-patch-to-br-int', 'ofport']),
|
'patch-fake-patch-to-br-int', 'ofport']),
|
||||||
mock.call('ovs-vsctl', ['get', 'Interface',
|
mock.call(['get', 'Interface',
|
||||||
'patch-fake-patch-to-br-int', 'ofport']),
|
'patch-fake-patch-to-br-int', 'ofport']),
|
||||||
mock.call('ovs-vsctl', ['get', 'Interface',
|
mock.call(['get', 'Interface',
|
||||||
'patch-fake-patch-to-br-int', 'ofport'])]
|
'patch-fake-patch-to-br-int', 'ofport'])]
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_has_calls(expected_calls)
|
self.mock_ovs_vsctl.ovs_vsctl.assert_has_calls(expected_calls)
|
||||||
|
|
||||||
@mock.patch.object(ovs_utils, 'get_bridge_flows')
|
@mock.patch.object(ovs_utils, 'get_bridge_flows')
|
||||||
def test_remove_extra_ovs_flows(self, mock_flows):
|
def test_remove_extra_ovs_flows(self, mock_flows):
|
||||||
@ -161,8 +159,8 @@ class TestOVS(test_base.TestCase):
|
|||||||
|
|
||||||
expected_del_flow = ('%s,ip,in_port=%s' % (self.cookie_id,
|
expected_del_flow = ('%s,ip,in_port=%s' % (self.cookie_id,
|
||||||
extra_port_iface))
|
extra_port_iface))
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_called_once_with(
|
self.mock_ovs_vsctl.ovs_ofctl.assert_called_once_with(
|
||||||
'ovs-ofctl', ['del-flows', self.bridge, expected_del_flow])
|
['del-flows', self.bridge, expected_del_flow])
|
||||||
mock_flows.assert_called_once_with(self.bridge, self.cookie_id)
|
mock_flows.assert_called_once_with(self.bridge, self.cookie_id)
|
||||||
|
|
||||||
def test_ensure_flow(self):
|
def test_ensure_flow(self):
|
||||||
@ -171,8 +169,8 @@ class TestOVS(test_base.TestCase):
|
|||||||
|
|
||||||
ovs_utils.ensure_flow(bridge, flow)
|
ovs_utils.ensure_flow(bridge, flow)
|
||||||
|
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_called_once_with(
|
self.mock_ovs_vsctl.ovs_ofctl.assert_called_once_with(
|
||||||
'ovs-ofctl', ['add-flow', bridge, flow])
|
['add-flow', bridge, flow])
|
||||||
|
|
||||||
@mock.patch.object(ovs_utils, 'get_device_port_at_ovs')
|
@mock.patch.object(ovs_utils, 'get_device_port_at_ovs')
|
||||||
@mock.patch.object(linux_net, 'get_interface_address')
|
@mock.patch.object(linux_net, 'get_interface_address')
|
||||||
@ -188,8 +186,9 @@ class TestOVS(test_base.TestCase):
|
|||||||
port_iface = '1'
|
port_iface = '1'
|
||||||
ovs_port_iface = '2'
|
ovs_port_iface = '2'
|
||||||
net = 'fake-net'
|
net = 'fake-net'
|
||||||
self.mock_ovs_vsctl.ovs_cmd.side_effect = (
|
self.mock_ovs_vsctl.ovs_vsctl.return_value = [
|
||||||
['%s\n%s\n' % (port, ovs_port)], None)
|
'%s\n%s\n' % (port, ovs_port)]
|
||||||
|
self.mock_ovs_vsctl.ovs_ofctl.return_value = None
|
||||||
mock_ofport.side_effect = (ovs_port_iface, port_iface)
|
mock_ofport.side_effect = (ovs_port_iface, port_iface)
|
||||||
|
|
||||||
# Invoke the method
|
# Invoke the method
|
||||||
@ -211,15 +210,14 @@ class TestOVS(test_base.TestCase):
|
|||||||
"ipv6_src={} actions=mod_dl_dst:{},{}output={}".format(
|
"ipv6_src={} actions=mod_dl_dst:{},{}output={}".format(
|
||||||
self.cookie, ovs_port_iface, self.mac, net, address,
|
self.cookie, ovs_port_iface, self.mac, net, address,
|
||||||
strip_vlan_opt, port_iface))
|
strip_vlan_opt, port_iface))
|
||||||
expected_calls = [
|
vsctl_expected_calls = [
|
||||||
mock.call('ovs-vsctl', ['list-ports', self.bridge]),
|
mock.call(['list-ports', self.bridge])]
|
||||||
mock.call('ovs-ofctl', ['add-flow', self.bridge, expected_flow])]
|
ofctl_expected_calls = [
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_has_calls(expected_calls)
|
mock.call(['add-flow', self.bridge, expected_flow])]
|
||||||
self.assertEqual(len(expected_calls),
|
self.mock_ovs_vsctl.ovs_vsctl.assert_has_calls(vsctl_expected_calls)
|
||||||
self.mock_ovs_vsctl.ovs_cmd.call_count)
|
self.mock_ovs_vsctl.ovs_ofctl.assert_has_calls(ofctl_expected_calls)
|
||||||
expected_calls_ofport = [mock.call(ovs_port), mock.call(port)]
|
expected_calls_ofport = [mock.call(ovs_port), mock.call(port)]
|
||||||
mock_ofport.assert_has_calls(expected_calls_ofport)
|
mock_ofport.assert_has_calls(expected_calls_ofport)
|
||||||
self.assertEqual(len(expected_calls_ofport), mock_ofport.call_count)
|
|
||||||
|
|
||||||
def test_ensure_evpn_ovs_flow_ipv4(self):
|
def test_ensure_evpn_ovs_flow_ipv4(self):
|
||||||
self._test_ensure_evpn_ovs_flow(ip_version=constants.IP_VERSION_4)
|
self._test_ensure_evpn_ovs_flow(ip_version=constants.IP_VERSION_4)
|
||||||
@ -237,21 +235,22 @@ class TestOVS(test_base.TestCase):
|
|||||||
|
|
||||||
def test_ensure_evpn_ovs_flow_no_ovs_ports(self):
|
def test_ensure_evpn_ovs_flow_no_ovs_ports(self):
|
||||||
port = 'non-patch-provnet-port'
|
port = 'non-patch-provnet-port'
|
||||||
self.mock_ovs_vsctl.ovs_cmd.return_value = [port]
|
self.mock_ovs_vsctl.ovs_vsctl.return_value = [port]
|
||||||
|
|
||||||
ret = ovs_utils.ensure_evpn_ovs_flow(
|
ret = ovs_utils.ensure_evpn_ovs_flow(
|
||||||
self.bridge, self.cookie, self.mac, port, 'fake-port-dst',
|
self.bridge, self.cookie, self.mac, port, 'fake-port-dst',
|
||||||
'fake-net')
|
'fake-net')
|
||||||
|
|
||||||
self.assertIsNone(ret)
|
self.assertIsNone(ret)
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_called_once_with(
|
self.mock_ovs_vsctl.ovs_vsctl.assert_called_once_with(
|
||||||
'ovs-vsctl', ['list-ports', self.bridge])
|
['list-ports', self.bridge])
|
||||||
|
|
||||||
@mock.patch.object(ovs_utils, 'get_device_port_at_ovs')
|
@mock.patch.object(ovs_utils, 'get_device_port_at_ovs')
|
||||||
def test_remove_evpn_router_ovs_flows(self, mock_ofport):
|
def test_remove_evpn_router_ovs_flows(self, mock_ofport):
|
||||||
ovs_port = constants.OVS_PATCH_PROVNET_PORT_PREFIX + 'fake-port'
|
ovs_port = constants.OVS_PATCH_PROVNET_PORT_PREFIX + 'fake-port'
|
||||||
ovs_port_iface = '1'
|
ovs_port_iface = '1'
|
||||||
self.mock_ovs_vsctl.ovs_cmd.side_effect = ([ovs_port], None, None)
|
self.mock_ovs_vsctl.ovs_vsctl.return_value = [ovs_port]
|
||||||
|
self.mock_ovs_vsctl.ovs_ofctl.side_effect = (None, None)
|
||||||
mock_ofport.return_value = ovs_port_iface
|
mock_ofport.return_value = ovs_port_iface
|
||||||
|
|
||||||
# Invoke the method
|
# Invoke the method
|
||||||
@ -263,26 +262,27 @@ class TestOVS(test_base.TestCase):
|
|||||||
expected_flow_v6 = '{},ipv6,in_port={},dl_src:{}'.format(
|
expected_flow_v6 = '{},ipv6,in_port={},dl_src:{}'.format(
|
||||||
self.cookie_id, ovs_port_iface, self.mac)
|
self.cookie_id, ovs_port_iface, self.mac)
|
||||||
|
|
||||||
expected_calls = [
|
vsctl_expected_calls = [
|
||||||
mock.call('ovs-vsctl', ['list-ports', self.bridge]),
|
mock.call(['list-ports', self.bridge])]
|
||||||
mock.call('ovs-ofctl', ['del-flows', self.bridge, expected_flow]),
|
ofctl_expected_calls = [
|
||||||
mock.call('ovs-ofctl', ['del-flows', self.bridge,
|
mock.call(['del-flows', self.bridge, expected_flow]),
|
||||||
expected_flow_v6])]
|
mock.call(['del-flows', self.bridge, expected_flow_v6])]
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_has_calls(expected_calls)
|
|
||||||
self.assertEqual(len(expected_calls),
|
self.mock_ovs_vsctl.ovs_vsctl.assert_has_calls(vsctl_expected_calls)
|
||||||
self.mock_ovs_vsctl.ovs_cmd.call_count)
|
self.mock_ovs_vsctl.ovs_ofctl.assert_has_calls(ofctl_expected_calls)
|
||||||
|
|
||||||
mock_ofport.assert_called_once_with(ovs_port)
|
mock_ofport.assert_called_once_with(ovs_port)
|
||||||
|
|
||||||
def test_remove_evpn_router_ovs_flows_no_ovs_port(self):
|
def test_remove_evpn_router_ovs_flows_no_ovs_port(self):
|
||||||
port = 'non-patch-provnet-port'
|
port = 'non-patch-provnet-port'
|
||||||
self.mock_ovs_vsctl.ovs_cmd.return_value = [port]
|
self.mock_ovs_vsctl.ovs_vsctl.return_value = [port]
|
||||||
|
|
||||||
ret = ovs_utils.remove_evpn_router_ovs_flows(
|
ret = ovs_utils.remove_evpn_router_ovs_flows(
|
||||||
self.bridge, self.cookie, self.mac)
|
self.bridge, self.cookie, self.mac)
|
||||||
|
|
||||||
self.assertIsNone(ret)
|
self.assertIsNone(ret)
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_called_once_with(
|
self.mock_ovs_vsctl.ovs_vsctl.assert_called_once_with(
|
||||||
'ovs-vsctl', ['list-ports', self.bridge])
|
['list-ports', self.bridge])
|
||||||
|
|
||||||
@mock.patch.object(ovs_utils, 'get_device_port_at_ovs')
|
@mock.patch.object(ovs_utils, 'get_device_port_at_ovs')
|
||||||
@mock.patch.object(linux_net, 'get_ip_version')
|
@mock.patch.object(linux_net, 'get_ip_version')
|
||||||
@ -293,7 +293,8 @@ class TestOVS(test_base.TestCase):
|
|||||||
net = 'fake-net'
|
net = 'fake-net'
|
||||||
mock_ip_version.return_value = ip_version
|
mock_ip_version.return_value = ip_version
|
||||||
mock_ofport.return_value = ovs_port_iface
|
mock_ofport.return_value = ovs_port_iface
|
||||||
self.mock_ovs_vsctl.ovs_cmd.side_effect = ([ovs_port], None)
|
self.mock_ovs_vsctl.ovs_vsctl.return_value = [ovs_port]
|
||||||
|
self.mock_ovs_vsctl.ovs_ofctl.side_effect = None
|
||||||
|
|
||||||
ovs_utils.remove_evpn_network_ovs_flow(
|
ovs_utils.remove_evpn_network_ovs_flow(
|
||||||
self.bridge, self.cookie, self.mac, net)
|
self.bridge, self.cookie, self.mac, net)
|
||||||
@ -305,12 +306,12 @@ class TestOVS(test_base.TestCase):
|
|||||||
expected_flow = ("{},ip,in_port={},dl_src:{},nw_src={}".format(
|
expected_flow = ("{},ip,in_port={},dl_src:{},nw_src={}".format(
|
||||||
self.cookie_id, ovs_port_iface, self.mac, net))
|
self.cookie_id, ovs_port_iface, self.mac, net))
|
||||||
|
|
||||||
expected_calls = [
|
vsctl_expected_calls = [
|
||||||
mock.call('ovs-vsctl', ['list-ports', self.bridge]),
|
mock.call(['list-ports', self.bridge])]
|
||||||
mock.call('ovs-ofctl', ['del-flows', self.bridge, expected_flow])]
|
ofctl_expected_calls = [
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_has_calls(expected_calls)
|
mock.call(['del-flows', self.bridge, expected_flow])]
|
||||||
self.assertEqual(len(expected_calls),
|
self.mock_ovs_vsctl.ovs_vsctl.assert_has_calls(vsctl_expected_calls)
|
||||||
self.mock_ovs_vsctl.ovs_cmd.call_count)
|
self.mock_ovs_vsctl.ovs_ofctl.assert_has_calls(ofctl_expected_calls)
|
||||||
mock_ip_version.assert_called_once_with(net)
|
mock_ip_version.assert_called_once_with(net)
|
||||||
|
|
||||||
def test_remove_evpn_network_ovs_flow_ipv4(self):
|
def test_remove_evpn_network_ovs_flow_ipv4(self):
|
||||||
@ -323,13 +324,13 @@ class TestOVS(test_base.TestCase):
|
|||||||
|
|
||||||
def test_remove_evpn_network_ovs_flow_no_ovs_port(self):
|
def test_remove_evpn_network_ovs_flow_no_ovs_port(self):
|
||||||
port = 'non-patch-provnet-port'
|
port = 'non-patch-provnet-port'
|
||||||
self.mock_ovs_vsctl.ovs_cmd.return_value = [port]
|
self.mock_ovs_vsctl.ovs_vsctl.return_value = [port]
|
||||||
|
|
||||||
ovs_utils.remove_evpn_network_ovs_flow(
|
ovs_utils.remove_evpn_network_ovs_flow(
|
||||||
self.bridge, self.cookie, self.mac, 'fake-net')
|
self.bridge, self.cookie, self.mac, 'fake-net')
|
||||||
|
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_called_once_with(
|
self.mock_ovs_vsctl.ovs_vsctl.assert_called_once_with(
|
||||||
'ovs-vsctl', ['list-ports', self.bridge])
|
['list-ports', self.bridge])
|
||||||
|
|
||||||
def _test_add_device_to_ovs_bridge(self, vlan_tag=False):
|
def _test_add_device_to_ovs_bridge(self, vlan_tag=False):
|
||||||
device = 'ethX'
|
device = 'ethX'
|
||||||
@ -341,8 +342,7 @@ class TestOVS(test_base.TestCase):
|
|||||||
if vlan_tag:
|
if vlan_tag:
|
||||||
expected_args.append('tag=%s' % vtag)
|
expected_args.append('tag=%s' % vtag)
|
||||||
|
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_called_once_with(
|
self.mock_ovs_vsctl.ovs_vsctl.assert_called_once_with(expected_args)
|
||||||
'ovs-vsctl', expected_args)
|
|
||||||
|
|
||||||
def test_add_device_to_ovs_bridge(self):
|
def test_add_device_to_ovs_bridge(self):
|
||||||
self._test_add_device_to_ovs_bridge()
|
self._test_add_device_to_ovs_bridge()
|
||||||
@ -361,8 +361,7 @@ class TestOVS(test_base.TestCase):
|
|||||||
expected_args.append(br)
|
expected_args.append(br)
|
||||||
expected_args.append(device)
|
expected_args.append(device)
|
||||||
|
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_called_once_with(
|
self.mock_ovs_vsctl.ovs_vsctl.assert_called_once_with(expected_args)
|
||||||
'ovs-vsctl', expected_args)
|
|
||||||
|
|
||||||
def test_del_device_from_ovs_bridge(self):
|
def test_del_device_from_ovs_bridge(self):
|
||||||
self._test_del_device_from_ovs_bridge()
|
self._test_del_device_from_ovs_bridge()
|
||||||
@ -379,8 +378,8 @@ class TestOVS(test_base.TestCase):
|
|||||||
|
|
||||||
expected_flow = ('{},priority=1000,ip,dl_src=fa:16:3e:15:9e:f0,'
|
expected_flow = ('{},priority=1000,ip,dl_src=fa:16:3e:15:9e:f0,'
|
||||||
'nw_src=20.0.0.0/24'.format(self.cookie_id))
|
'nw_src=20.0.0.0/24'.format(self.cookie_id))
|
||||||
self.mock_ovs_vsctl.ovs_cmd.assert_called_once_with(
|
self.mock_ovs_vsctl.ovs_ofctl.assert_called_once_with(
|
||||||
'ovs-ofctl', ['--strict', 'del-flows', self.bridge, expected_flow])
|
['--strict', 'del-flows', self.bridge, expected_flow])
|
||||||
|
|
||||||
def test_get_flow_info(self):
|
def test_get_flow_info(self):
|
||||||
flow = ('cookie=0x3e6, duration=11.647s, table=0, n_packets=0, '
|
flow = ('cookie=0x3e6, duration=11.647s, table=0, n_packets=0, '
|
||||||
|
@ -38,47 +38,68 @@ class TestPrivilegedOvsVsctl(test_base.TestCase):
|
|||||||
# Mock processutils.execute()
|
# Mock processutils.execute()
|
||||||
self.mock_exc = mock.patch.object(processutils, 'execute').start()
|
self.mock_exc = mock.patch.object(processutils, 'execute').start()
|
||||||
|
|
||||||
def test_ovs_cmd(self):
|
def test_ovs_vsctl(self):
|
||||||
ovs_vsctl.ovs_cmd(
|
ovs_vsctl.ovs_vsctl(
|
||||||
'ovs-vsctl', ['--if-exists', 'del-port', 'fake-port'])
|
['--if-exists', 'del-port', 'fake-port'])
|
||||||
self.mock_exc.assert_called_once_with(
|
self.mock_exc.assert_called_once_with(
|
||||||
'ovs-vsctl', '--if-exists', 'del-port', 'fake-port')
|
'ovs-vsctl', '--if-exists', 'del-port', 'fake-port')
|
||||||
|
|
||||||
def test_ovs_cmd_timeout(self):
|
def test_ovs_ofctl(self):
|
||||||
ovs_vsctl.ovs_cmd(
|
ovs_vsctl.ovs_ofctl(
|
||||||
'ovs-vsctl', ['--if-exists', 'del-port', 'fake-port'], timeout=10)
|
['dump-flows', 'dummy-br'])
|
||||||
|
self.mock_exc.assert_called_once_with(
|
||||||
|
'ovs-ofctl', 'dump-flows', 'dummy-br')
|
||||||
|
|
||||||
|
def test_ovs_vsctl_timeout(self):
|
||||||
|
ovs_vsctl.ovs_vsctl(
|
||||||
|
['--if-exists', 'del-port', 'fake-port'], timeout=10)
|
||||||
self.mock_exc.assert_called_once_with(
|
self.mock_exc.assert_called_once_with(
|
||||||
'ovs-vsctl', '--timeout=10', '--if-exists', 'del-port',
|
'ovs-vsctl', '--timeout=10', '--if-exists', 'del-port',
|
||||||
'fake-port')
|
'fake-port')
|
||||||
|
|
||||||
def test_ovs_cmd_fallback_OF_version(self):
|
def test_ovs_ofctl_timeout(self):
|
||||||
|
ovs_vsctl.ovs_ofctl(
|
||||||
|
['dump-flows', 'dummy-br'], timeout=10)
|
||||||
|
self.mock_exc.assert_called_once_with(
|
||||||
|
'ovs-ofctl', '--timeout=10', 'dump-flows', 'dummy-br')
|
||||||
|
|
||||||
|
def test_ovs_ofctl_fallback_OF_version(self):
|
||||||
|
# fallback only applies to ovs-ofctl command
|
||||||
self.mock_exc.side_effect = (
|
self.mock_exc.side_effect = (
|
||||||
processutils.ProcessExecutionError(), None)
|
processutils.ProcessExecutionError(), None)
|
||||||
ovs_vsctl.ovs_cmd(
|
ovs_vsctl.ovs_ofctl(
|
||||||
'ovs-vsctl', ['--if-exists', 'del-port', 'fake-port'])
|
['--strict', 'del-flows', 'br-ex', 'dummy-flow'])
|
||||||
|
|
||||||
calls = [mock.call('ovs-vsctl', '--if-exists', 'del-port',
|
calls = [mock.call('ovs-ofctl', '--strict', 'del-flows',
|
||||||
'fake-port'),
|
'br-ex', 'dummy-flow'),
|
||||||
mock.call('ovs-vsctl', '--if-exists', 'del-port',
|
mock.call('ovs-ofctl', '--strict', 'del-flows',
|
||||||
'fake-port', '-O', 'OpenFlow13')]
|
'br-ex', 'dummy-flow', '-O', 'OpenFlow13')]
|
||||||
self.mock_exc.assert_has_calls(calls)
|
self.mock_exc.assert_has_calls(calls)
|
||||||
|
|
||||||
def test_ovs_cmd_exception(self):
|
def test_ovs_vsctl_process_execution_error_no_fallback(self):
|
||||||
self.mock_exc.side_effect = FakeException()
|
# fallback does not apply to ovs-vsctl command
|
||||||
|
self.mock_exc.side_effect = processutils.ProcessExecutionError()
|
||||||
self.assertRaises(
|
self.assertRaises(
|
||||||
FakeException, ovs_vsctl.ovs_cmd, 'ovs-vsctl',
|
processutils.ProcessExecutionError, ovs_vsctl.ovs_vsctl,
|
||||||
['--if-exists', 'del-port', 'fake-port'])
|
['--if-exists', 'del-port', 'fake-port'])
|
||||||
self.mock_exc.assert_called_once_with(
|
self.mock_exc.assert_called_once_with(
|
||||||
'ovs-vsctl', '--if-exists', 'del-port', 'fake-port')
|
'ovs-vsctl', '--if-exists', 'del-port', 'fake-port')
|
||||||
|
|
||||||
|
def test_ovs_ofctl_exception(self):
|
||||||
|
self.mock_exc.side_effect = FakeException()
|
||||||
|
self.assertRaises(
|
||||||
|
FakeException, ovs_vsctl.ovs_ofctl,
|
||||||
|
['add-flow', 'br-ex', 'dummy-flow'])
|
||||||
|
self.mock_exc.assert_called_once_with(
|
||||||
|
'ovs-ofctl', 'add-flow', 'br-ex', 'dummy-flow')
|
||||||
|
|
||||||
def test_ovs_cmd_fallback_exception(self):
|
def test_ovs_cmd_fallback_exception(self):
|
||||||
self.mock_exc.side_effect = (
|
self.mock_exc.side_effect = (
|
||||||
processutils.ProcessExecutionError(), FakeException())
|
processutils.ProcessExecutionError(), FakeException())
|
||||||
self.assertRaises(
|
self.assertRaises(
|
||||||
FakeException, ovs_vsctl.ovs_cmd, 'ovs-vsctl',
|
FakeException, ovs_vsctl.ovs_ofctl,
|
||||||
['--if-exists', 'del-port', 'fake-port'])
|
['add-flow', 'br-ex', 'dummy-flow'])
|
||||||
calls = [mock.call('ovs-vsctl', '--if-exists', 'del-port',
|
calls = [mock.call('ovs-ofctl', 'add-flow', 'br-ex', 'dummy-flow'),
|
||||||
'fake-port'),
|
mock.call('ovs-ofctl', 'add-flow', 'br-ex', 'dummy-flow',
|
||||||
mock.call('ovs-vsctl', '--if-exists', 'del-port',
|
'-O', 'OpenFlow13')]
|
||||||
'fake-port', '-O', 'OpenFlow13')]
|
|
||||||
self.mock_exc.assert_has_calls(calls)
|
self.mock_exc.assert_has_calls(calls)
|
||||||
|
Loading…
Reference in New Issue
Block a user