Fix l2 pop doesn't propagate ip address updates
Propagates ip address changes when an ip address is : added, removed, or changed. Add a new rpc call for the updates of forwarding informations. Fixes: Bug #1234137 Change-Id: Ib5b971bd02f20a0ea73f88ce9685e944226bb5a2
This commit is contained in:
parent
cc0ed67c01
commit
40779c5b0f
@ -37,6 +37,11 @@ class L2populationRpcCallBackMixin(object):
|
|||||||
if not host or host == cfg.CONF.host:
|
if not host or host == cfg.CONF.host:
|
||||||
self.fdb_remove(context, fdb_entries)
|
self.fdb_remove(context, fdb_entries)
|
||||||
|
|
||||||
|
@log.log
|
||||||
|
def update_fdb_entries(self, context, fdb_entries, host=None):
|
||||||
|
if not host or host == cfg.CONF.host:
|
||||||
|
self.fdb_update(context, fdb_entries)
|
||||||
|
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def fdb_add(self, context, fdb_entries):
|
def fdb_add(self, context, fdb_entries):
|
||||||
pass
|
pass
|
||||||
@ -44,3 +49,7 @@ class L2populationRpcCallBackMixin(object):
|
|||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def fdb_remove(self, context, fdb_entries):
|
def fdb_remove(self, context, fdb_entries):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
@abc.abstractmethod
|
||||||
|
def fdb_update(self, context, fdb_entries):
|
||||||
|
pass
|
||||||
|
@ -714,6 +714,40 @@ class LinuxBridgeRpcCallbacks(sg_rpc.SecurityGroupAgentRpcCallbackMixin,
|
|||||||
ports,
|
ports,
|
||||||
interface)
|
interface)
|
||||||
|
|
||||||
|
def _fdb_chg_ip(self, context, fdb_entries):
|
||||||
|
LOG.debug(_("update chg_ip received"))
|
||||||
|
for network_id, agent_ports in fdb_entries.items():
|
||||||
|
segment = self.agent.br_mgr.network_map.get(network_id)
|
||||||
|
if not segment:
|
||||||
|
return
|
||||||
|
|
||||||
|
if segment.network_type != lconst.TYPE_VXLAN:
|
||||||
|
return
|
||||||
|
|
||||||
|
interface = self.agent.br_mgr.get_vxlan_device_name(
|
||||||
|
segment.segmentation_id)
|
||||||
|
|
||||||
|
for agent_ip, state in agent_ports.items():
|
||||||
|
if agent_ip == self.agent.br_mgr.local_ip:
|
||||||
|
continue
|
||||||
|
|
||||||
|
after = state.get('after')
|
||||||
|
for mac, ip in after:
|
||||||
|
self.agent.br_mgr.add_fdb_ip_entry(mac, ip, interface)
|
||||||
|
|
||||||
|
before = state.get('before')
|
||||||
|
for mac, ip in before:
|
||||||
|
self.agent.br_mgr.remove_fdb_ip_entry(mac, ip, interface)
|
||||||
|
|
||||||
|
def fdb_update(self, context, fdb_entries):
|
||||||
|
LOG.debug(_("fdb_update received"))
|
||||||
|
for action, values in fdb_entries.items():
|
||||||
|
method = '_fdb_' + action
|
||||||
|
if not hasattr(self, method):
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
getattr(self, method)(context, values)
|
||||||
|
|
||||||
def create_rpc_dispatcher(self):
|
def create_rpc_dispatcher(self):
|
||||||
'''Get the rpc dispatcher for this manager.
|
'''Get the rpc dispatcher for this manager.
|
||||||
|
|
||||||
|
@ -36,6 +36,7 @@ class L2populationMechanismDriver(api.MechanismDriver,
|
|||||||
|
|
||||||
def initialize(self):
|
def initialize(self):
|
||||||
LOG.debug(_("Experimental L2 population driver"))
|
LOG.debug(_("Experimental L2 population driver"))
|
||||||
|
self.rpc_ctx = n_context.get_admin_context_without_session()
|
||||||
|
|
||||||
def _get_port_fdb_entries(self, port):
|
def _get_port_fdb_entries(self, port):
|
||||||
return [[port['mac_address'],
|
return [[port['mac_address'],
|
||||||
@ -45,31 +46,64 @@ class L2populationMechanismDriver(api.MechanismDriver,
|
|||||||
self.remove_fdb_entries = self._update_port_down(context)
|
self.remove_fdb_entries = self._update_port_down(context)
|
||||||
|
|
||||||
def delete_port_postcommit(self, context):
|
def delete_port_postcommit(self, context):
|
||||||
self._notify_remove_fdb_entries(context,
|
|
||||||
self.remove_fdb_entries)
|
|
||||||
|
|
||||||
def _notify_remove_fdb_entries(self, context, fdb_entries):
|
|
||||||
rpc_ctx = n_context.get_admin_context_without_session()
|
|
||||||
l2pop_rpc.L2populationAgentNotify.remove_fdb_entries(
|
l2pop_rpc.L2populationAgentNotify.remove_fdb_entries(
|
||||||
rpc_ctx, fdb_entries)
|
self.rpc_ctx, self.remove_fdb_entries)
|
||||||
|
|
||||||
|
def _get_diff_ips(self, orig, port):
|
||||||
|
orig_ips = set([ip['ip_address'] for ip in orig['fixed_ips']])
|
||||||
|
port_ips = set([ip['ip_address'] for ip in port['fixed_ips']])
|
||||||
|
|
||||||
|
# check if an ip has been added or removed
|
||||||
|
orig_chg_ips = orig_ips.difference(port_ips)
|
||||||
|
port_chg_ips = port_ips.difference(orig_ips)
|
||||||
|
|
||||||
|
if orig_chg_ips or port_chg_ips:
|
||||||
|
return orig_chg_ips, port_chg_ips
|
||||||
|
|
||||||
|
def _fixed_ips_changed(self, context, orig, port):
|
||||||
|
diff_ips = self._get_diff_ips(orig, port)
|
||||||
|
if not diff_ips:
|
||||||
|
return
|
||||||
|
orig_ips, port_ips = diff_ips
|
||||||
|
|
||||||
|
port_infos = self._get_port_infos(context, orig)
|
||||||
|
if not port_infos:
|
||||||
|
return
|
||||||
|
agent, agent_ip, segment, port_fdb_entries = port_infos
|
||||||
|
|
||||||
|
orig_mac_ip = [[port['mac_address'], ip] for ip in orig_ips]
|
||||||
|
port_mac_ip = [[port['mac_address'], ip] for ip in port_ips]
|
||||||
|
|
||||||
|
upd_fdb_entries = {port['network_id']: {agent_ip: {}}}
|
||||||
|
|
||||||
|
ports = upd_fdb_entries[port['network_id']][agent_ip]
|
||||||
|
if orig_mac_ip:
|
||||||
|
ports['before'] = orig_mac_ip
|
||||||
|
|
||||||
|
if port_mac_ip:
|
||||||
|
ports['after'] = port_mac_ip
|
||||||
|
|
||||||
|
l2pop_rpc.L2populationAgentNotify.update_fdb_entries(
|
||||||
|
self.rpc_ctx, {'chg_ip': upd_fdb_entries})
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
def update_port_postcommit(self, context):
|
def update_port_postcommit(self, context):
|
||||||
port = context.current
|
port = context.current
|
||||||
orig = context.original
|
orig = context.original
|
||||||
|
|
||||||
if port['status'] == orig['status']:
|
if port['status'] == orig['status']:
|
||||||
return
|
self._fixed_ips_changed(context, orig, port)
|
||||||
|
|
||||||
if port['status'] == const.PORT_STATUS_ACTIVE:
|
if port['status'] == const.PORT_STATUS_ACTIVE:
|
||||||
self._update_port_up(context)
|
self._update_port_up(context)
|
||||||
elif port['status'] == const.PORT_STATUS_DOWN:
|
elif port['status'] == const.PORT_STATUS_DOWN:
|
||||||
fdb_entries = self._update_port_down(context)
|
fdb_entries = self._update_port_down(context)
|
||||||
self._notify_remove_fdb_entries(context, fdb_entries)
|
l2pop_rpc.L2populationAgentNotify.remove_fdb_entries(
|
||||||
|
self.rpc_ctx, fdb_entries)
|
||||||
|
|
||||||
def _update_port_up(self, context):
|
def _get_port_infos(self, context, port):
|
||||||
port_context = context.current
|
agent_host = port['binding:host_id']
|
||||||
network_id = port_context['network_id']
|
|
||||||
agent_host = port_context['binding:host_id']
|
|
||||||
if not agent_host:
|
if not agent_host:
|
||||||
return
|
return
|
||||||
|
|
||||||
@ -80,26 +114,39 @@ class L2populationMechanismDriver(api.MechanismDriver,
|
|||||||
|
|
||||||
agent_ip = self.get_agent_ip(agent)
|
agent_ip = self.get_agent_ip(agent)
|
||||||
if not agent_ip:
|
if not agent_ip:
|
||||||
LOG.warning(_("Unable to retrieve the tunelling ip of agent %s"),
|
LOG.warning(_("Unable to retrieve the agent ip, check the agent "
|
||||||
agent_host)
|
"configuration."))
|
||||||
return
|
return
|
||||||
|
|
||||||
segment = context.bound_segment
|
segment = context.bound_segment
|
||||||
if not segment:
|
if not segment:
|
||||||
LOG.warning(_("Port %(port)s updated by agent %(agent)s "
|
LOG.warning(_("Port %(port)s updated by agent %(agent)s "
|
||||||
"isn't bound to any segment"),
|
"isn't bound to any segment"),
|
||||||
{'port': port_context['id'], 'agent': agent.host})
|
{'port': port['id'], 'agent': agent})
|
||||||
return
|
return
|
||||||
|
|
||||||
tunnel_types = self.get_agent_tunnel_types(agent)
|
tunnel_types = self.get_agent_tunnel_types(agent)
|
||||||
if segment['network_type'] not in tunnel_types:
|
if segment['network_type'] not in tunnel_types:
|
||||||
return
|
return
|
||||||
|
|
||||||
|
fdb_entries = self._get_port_fdb_entries(port)
|
||||||
|
|
||||||
|
return agent, agent_ip, segment, fdb_entries
|
||||||
|
|
||||||
|
def _update_port_up(self, context):
|
||||||
|
port_context = context.current
|
||||||
|
port_infos = self._get_port_infos(context, port_context)
|
||||||
|
if not port_infos:
|
||||||
|
return
|
||||||
|
agent, agent_ip, segment, port_fdb_entries = port_infos
|
||||||
|
|
||||||
|
agent_host = port_context['binding:host_id']
|
||||||
|
network_id = port_context['network_id']
|
||||||
|
|
||||||
|
session = db_api.get_session()
|
||||||
agent_ports = self.get_agent_network_port_count(session, agent_host,
|
agent_ports = self.get_agent_network_port_count(session, agent_host,
|
||||||
network_id)
|
network_id)
|
||||||
|
|
||||||
rpc_ctx = n_context.get_admin_context_without_session()
|
|
||||||
|
|
||||||
other_fdb_entries = {network_id:
|
other_fdb_entries = {network_id:
|
||||||
{'segment_id': segment['segmentation_id'],
|
{'segment_id': segment['segmentation_id'],
|
||||||
'network_type': segment['network_type'],
|
'network_type': segment['network_type'],
|
||||||
@ -138,45 +185,25 @@ class L2populationMechanismDriver(api.MechanismDriver,
|
|||||||
|
|
||||||
if ports.keys():
|
if ports.keys():
|
||||||
l2pop_rpc.L2populationAgentNotify.add_fdb_entries(
|
l2pop_rpc.L2populationAgentNotify.add_fdb_entries(
|
||||||
rpc_ctx, agent_fdb_entries, agent_host)
|
self.rpc_ctx, agent_fdb_entries, agent_host)
|
||||||
|
|
||||||
# Notify other agents to add fdb rule for current port
|
# Notify other agents to add fdb rule for current port
|
||||||
fdb_entries = self._get_port_fdb_entries(port_context)
|
other_fdb_entries[network_id]['ports'][agent_ip] += port_fdb_entries
|
||||||
other_fdb_entries[network_id]['ports'][agent_ip] += fdb_entries
|
|
||||||
|
|
||||||
l2pop_rpc.L2populationAgentNotify.add_fdb_entries(rpc_ctx,
|
l2pop_rpc.L2populationAgentNotify.add_fdb_entries(self.rpc_ctx,
|
||||||
other_fdb_entries)
|
other_fdb_entries)
|
||||||
|
|
||||||
def _update_port_down(self, context):
|
def _update_port_down(self, context):
|
||||||
port_context = context.current
|
port_context = context.current
|
||||||
network_id = port_context['network_id']
|
port_infos = self._get_port_infos(context, port_context)
|
||||||
|
if not port_infos:
|
||||||
|
return
|
||||||
|
agent, agent_ip, segment, port_fdb_entries = port_infos
|
||||||
|
|
||||||
agent_host = port_context['binding:host_id']
|
agent_host = port_context['binding:host_id']
|
||||||
if not agent_host:
|
network_id = port_context['network_id']
|
||||||
return
|
|
||||||
|
|
||||||
session = db_api.get_session()
|
session = db_api.get_session()
|
||||||
agent = self.get_agent_by_host(session, agent_host)
|
|
||||||
if not agent:
|
|
||||||
return
|
|
||||||
|
|
||||||
agent_ip = self.get_agent_ip(agent)
|
|
||||||
if not agent_ip:
|
|
||||||
LOG.warning(_("Unable to retrieve the agent ip, check the agent "
|
|
||||||
"configuration."))
|
|
||||||
return
|
|
||||||
|
|
||||||
segment = context.bound_segment
|
|
||||||
if not segment:
|
|
||||||
LOG.warning(_("Port %(port)s updated by agent %(agent)s "
|
|
||||||
"isn't bound to any segment"),
|
|
||||||
{'port': port_context['id'], 'agent': agent})
|
|
||||||
return
|
|
||||||
|
|
||||||
tunnel_types = self.get_agent_tunnel_types(agent)
|
|
||||||
if segment['network_type'] not in tunnel_types:
|
|
||||||
return
|
|
||||||
|
|
||||||
agent_ports = self.get_agent_network_port_count(session, agent_host,
|
agent_ports = self.get_agent_network_port_count(session, agent_host,
|
||||||
network_id)
|
network_id)
|
||||||
|
|
||||||
|
@ -76,4 +76,13 @@ class L2populationAgentNotifyAPI(proxy.RpcProxy):
|
|||||||
self._notification_fanout(context, 'remove_fdb_entries',
|
self._notification_fanout(context, 'remove_fdb_entries',
|
||||||
fdb_entries)
|
fdb_entries)
|
||||||
|
|
||||||
|
def update_fdb_entries(self, context, fdb_entries, host=None):
|
||||||
|
if fdb_entries:
|
||||||
|
if host:
|
||||||
|
self._notification_host(context, 'update_fdb_entries',
|
||||||
|
fdb_entries, host)
|
||||||
|
else:
|
||||||
|
self._notification_fanout(context, 'update_fdb_entries',
|
||||||
|
fdb_entries)
|
||||||
|
|
||||||
L2populationAgentNotify = L2populationAgentNotifyAPI()
|
L2populationAgentNotify = L2populationAgentNotifyAPI()
|
||||||
|
@ -412,6 +412,15 @@ class OVSNeutronAgent(sg_rpc.SecurityGroupAgentRpcCallbackMixin,
|
|||||||
dl_vlan=lvm.vlan,
|
dl_vlan=lvm.vlan,
|
||||||
dl_dst=port_info[0])
|
dl_dst=port_info[0])
|
||||||
|
|
||||||
|
def fdb_update(self, context, fdb_entries):
|
||||||
|
LOG.debug(_("fdb_update received"))
|
||||||
|
for action, values in fdb_entries.items():
|
||||||
|
method = '_fdb_' + action
|
||||||
|
if not hasattr(self, method):
|
||||||
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
getattr(self, method)(context, values)
|
||||||
|
|
||||||
def create_rpc_dispatcher(self):
|
def create_rpc_dispatcher(self):
|
||||||
'''Get the rpc dispatcher for this manager.
|
'''Get the rpc dispatcher for this manager.
|
||||||
|
|
||||||
|
@ -878,3 +878,26 @@ class TestLinuxBridgeRpcCallbacks(base.BaseTestCase):
|
|||||||
check_exit_code=False),
|
check_exit_code=False),
|
||||||
]
|
]
|
||||||
execute_fn.assert_has_calls(expected)
|
execute_fn.assert_has_calls(expected)
|
||||||
|
|
||||||
|
def test_fdb_update_chg_ip(self):
|
||||||
|
fdb_entries = {'chg_ip':
|
||||||
|
{'net_id':
|
||||||
|
{'agent_ip':
|
||||||
|
{'before': [['port_mac', 'port_ip_1']],
|
||||||
|
'after': [['port_mac', 'port_ip_2']]}}}}
|
||||||
|
|
||||||
|
with mock.patch.object(utils, 'execute',
|
||||||
|
return_value='') as execute_fn:
|
||||||
|
self.lb_rpc.fdb_update(None, fdb_entries)
|
||||||
|
|
||||||
|
expected = [
|
||||||
|
mock.call(['ip', 'neigh', 'add', 'port_ip_2', 'lladdr',
|
||||||
|
'port_mac', 'dev', 'vxlan-1', 'nud', 'permanent'],
|
||||||
|
root_helper=self.root_helper,
|
||||||
|
check_exit_code=False),
|
||||||
|
mock.call(['ip', 'neigh', 'del', 'port_ip_1', 'lladdr',
|
||||||
|
'port_mac', 'dev', 'vxlan-1'],
|
||||||
|
root_helper=self.root_helper,
|
||||||
|
check_exit_code=False)
|
||||||
|
]
|
||||||
|
execute_fn.assert_has_calls(expected)
|
||||||
|
@ -406,3 +406,80 @@ class TestL2PopulationRpcTestCase(test_plugin.NeutronDbPluginV2TestCase):
|
|||||||
|
|
||||||
self.mock_fanout.assert_any_call(
|
self.mock_fanout.assert_any_call(
|
||||||
mock.ANY, expected, topic=self.fanout_topic)
|
mock.ANY, expected, topic=self.fanout_topic)
|
||||||
|
|
||||||
|
def test_fixed_ips_changed(self):
|
||||||
|
self._register_ml2_agents()
|
||||||
|
|
||||||
|
with self.subnet(network=self._network) as subnet:
|
||||||
|
host_arg = {portbindings.HOST_ID: HOST}
|
||||||
|
with self.port(subnet=subnet, cidr='10.0.0.0/24',
|
||||||
|
arg_list=(portbindings.HOST_ID,),
|
||||||
|
**host_arg) as port1:
|
||||||
|
p1 = port1['port']
|
||||||
|
|
||||||
|
self.mock_fanout.reset_mock()
|
||||||
|
|
||||||
|
data = {'port': {'fixed_ips': [{'ip_address': '10.0.0.2'},
|
||||||
|
{'ip_address': '10.0.0.10'}]}}
|
||||||
|
req = self.new_update_request('ports', data, p1['id'])
|
||||||
|
res = self.deserialize(self.fmt, req.get_response(self.api))
|
||||||
|
ips = res['port']['fixed_ips']
|
||||||
|
self.assertEqual(len(ips), 2)
|
||||||
|
|
||||||
|
add_expected = {'args':
|
||||||
|
{'fdb_entries':
|
||||||
|
{'chg_ip':
|
||||||
|
{p1['network_id']:
|
||||||
|
{'20.0.0.1':
|
||||||
|
{'after': [[p1['mac_address'],
|
||||||
|
'10.0.0.10']]}}}}},
|
||||||
|
'namespace': None,
|
||||||
|
'method': 'update_fdb_entries'}
|
||||||
|
|
||||||
|
self.mock_fanout.assert_any_call(
|
||||||
|
mock.ANY, add_expected, topic=self.fanout_topic)
|
||||||
|
|
||||||
|
self.mock_fanout.reset_mock()
|
||||||
|
|
||||||
|
data = {'port': {'fixed_ips': [{'ip_address': '10.0.0.2'},
|
||||||
|
{'ip_address': '10.0.0.16'}]}}
|
||||||
|
req = self.new_update_request('ports', data, p1['id'])
|
||||||
|
res = self.deserialize(self.fmt, req.get_response(self.api))
|
||||||
|
ips = res['port']['fixed_ips']
|
||||||
|
self.assertEqual(len(ips), 2)
|
||||||
|
|
||||||
|
upd_expected = {'args':
|
||||||
|
{'fdb_entries':
|
||||||
|
{'chg_ip':
|
||||||
|
{p1['network_id']:
|
||||||
|
{'20.0.0.1':
|
||||||
|
{'before': [[p1['mac_address'],
|
||||||
|
'10.0.0.10']],
|
||||||
|
'after': [[p1['mac_address'],
|
||||||
|
'10.0.0.16']]}}}}},
|
||||||
|
'namespace': None,
|
||||||
|
'method': 'update_fdb_entries'}
|
||||||
|
|
||||||
|
self.mock_fanout.assert_any_call(
|
||||||
|
mock.ANY, upd_expected, topic=self.fanout_topic)
|
||||||
|
|
||||||
|
self.mock_fanout.reset_mock()
|
||||||
|
|
||||||
|
data = {'port': {'fixed_ips': [{'ip_address': '10.0.0.16'}]}}
|
||||||
|
req = self.new_update_request('ports', data, p1['id'])
|
||||||
|
res = self.deserialize(self.fmt, req.get_response(self.api))
|
||||||
|
ips = res['port']['fixed_ips']
|
||||||
|
self.assertEqual(len(ips), 1)
|
||||||
|
|
||||||
|
del_expected = {'args':
|
||||||
|
{'fdb_entries':
|
||||||
|
{'chg_ip':
|
||||||
|
{p1['network_id']:
|
||||||
|
{'20.0.0.1':
|
||||||
|
{'before': [[p1['mac_address'],
|
||||||
|
'10.0.0.2']]}}}}},
|
||||||
|
'namespace': None,
|
||||||
|
'method': 'update_fdb_entries'}
|
||||||
|
|
||||||
|
self.mock_fanout.assert_any_call(
|
||||||
|
mock.ANY, del_expected, topic=self.fanout_topic)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user