Fix race in get_network(s) in OVS plugin

Load network bindings eagerly with networks.
Otherwise a different db query could try to fetch network bindings
for already deleted networks. The issue is reproducible with
concurrent tempest network API tests.

Closes-Bug: 1263686
Change-Id: I0521ab162220c66a62491ff8e7e4a9f6d7bef6c4
This commit is contained in:
Eugene Nikanorov 2013-12-24 15:08:22 +04:00
parent 1e302cc250
commit 84100792ae
3 changed files with 22 additions and 10 deletions

View File

@ -55,6 +55,7 @@ def add_network_binding(session, network_id, network_type,
physical_network, physical_network,
segmentation_id) segmentation_id)
session.add(binding) session.add(binding)
return binding
def sync_vlan_allocations(network_vlan_ranges): def sync_vlan_allocations(network_vlan_ranges):

View File

@ -20,7 +20,9 @@
from sqlalchemy import Boolean, Column, ForeignKey, Integer, String from sqlalchemy import Boolean, Column, ForeignKey, Integer, String
from sqlalchemy.schema import UniqueConstraint from sqlalchemy.schema import UniqueConstraint
from neutron.db import models_v2
from neutron.db.models_v2 import model_base from neutron.db.models_v2 import model_base
from sqlalchemy import orm
class VlanAllocation(model_base.BASEV2): class VlanAllocation(model_base.BASEV2):
@ -70,6 +72,11 @@ class NetworkBinding(model_base.BASEV2):
physical_network = Column(String(64)) physical_network = Column(String(64))
segmentation_id = Column(Integer) # tunnel_id or vlan_id segmentation_id = Column(Integer) # tunnel_id or vlan_id
network = orm.relationship(
models_v2.Network,
backref=orm.backref("binding", lazy='joined',
uselist=False, cascade='delete'))
def __init__(self, network_id, network_type, physical_network, def __init__(self, network_id, network_type, physical_network,
segmentation_id): segmentation_id):
self.network_id = network_id self.network_id = network_id

View File

@ -290,6 +290,9 @@ class OVSNeutronPluginV2(db_base_plugin_v2.NeutronDbPluginV2,
self._aliases = aliases self._aliases = aliases
return self._aliases return self._aliases
db_base_plugin_v2.NeutronDbPluginV2.register_dict_extend_funcs(
attributes.NETWORKS, ['_extend_network_dict_provider_ovs'])
def __init__(self, configfile=None): def __init__(self, configfile=None):
self.base_binding_dict = { self.base_binding_dict = {
portbindings.VIF_TYPE: portbindings.VIF_TYPE_OVS, portbindings.VIF_TYPE: portbindings.VIF_TYPE_OVS,
@ -374,9 +377,11 @@ class OVSNeutronPluginV2(db_base_plugin_v2.NeutronDbPluginV2,
sys.exit(1) sys.exit(1)
LOG.info(_("Tunnel ID ranges: %s"), self.tunnel_id_ranges) LOG.info(_("Tunnel ID ranges: %s"), self.tunnel_id_ranges)
def _extend_network_dict_provider(self, context, network): def _extend_network_dict_provider_ovs(self, network, net_db,
binding = ovs_db_v2.get_network_binding(context.session, net_binding=None):
network['id']) # this method used in two cases: when binding is provided explicitly
# and when it is a part of db model object
binding = net_db.binding if net_db else net_binding
network[provider.NETWORK_TYPE] = binding.network_type network[provider.NETWORK_TYPE] = binding.network_type
if binding.network_type in constants.TUNNEL_NETWORK_TYPES: if binding.network_type in constants.TUNNEL_NETWORK_TYPES:
network[provider.PHYSICAL_NETWORK] = None network[provider.PHYSICAL_NETWORK] = None
@ -501,11 +506,14 @@ class OVSNeutronPluginV2(db_base_plugin_v2.NeutronDbPluginV2,
# no reservation needed for TYPE_LOCAL # no reservation needed for TYPE_LOCAL
net = super(OVSNeutronPluginV2, self).create_network(context, net = super(OVSNeutronPluginV2, self).create_network(context,
network) network)
ovs_db_v2.add_network_binding(session, net['id'], network_type, binding = ovs_db_v2.add_network_binding(session, net['id'],
physical_network, segmentation_id) network_type,
physical_network,
segmentation_id)
self._process_l3_create(context, net, network['network']) self._process_l3_create(context, net, network['network'])
self._extend_network_dict_provider(context, net) # passing None as db model to use binding object
self._extend_network_dict_provider_ovs(net, None, binding)
# note - exception will rollback entire transaction # note - exception will rollback entire transaction
LOG.debug(_("Created network: %s"), net['id']) LOG.debug(_("Created network: %s"), net['id'])
return net return net
@ -518,7 +526,6 @@ class OVSNeutronPluginV2(db_base_plugin_v2.NeutronDbPluginV2,
net = super(OVSNeutronPluginV2, self).update_network(context, id, net = super(OVSNeutronPluginV2, self).update_network(context, id,
network) network)
self._process_l3_update(context, net, network['network']) self._process_l3_update(context, net, network['network'])
self._extend_network_dict_provider(context, net)
return net return net
def delete_network(self, context, id): def delete_network(self, context, id):
@ -543,7 +550,6 @@ class OVSNeutronPluginV2(db_base_plugin_v2.NeutronDbPluginV2,
with session.begin(subtransactions=True): with session.begin(subtransactions=True):
net = super(OVSNeutronPluginV2, self).get_network(context, net = super(OVSNeutronPluginV2, self).get_network(context,
id, None) id, None)
self._extend_network_dict_provider(context, net)
return self._fields(net, fields) return self._fields(net, fields)
def get_networks(self, context, filters=None, fields=None, def get_networks(self, context, filters=None, fields=None,
@ -554,8 +560,6 @@ class OVSNeutronPluginV2(db_base_plugin_v2.NeutronDbPluginV2,
nets = super(OVSNeutronPluginV2, nets = super(OVSNeutronPluginV2,
self).get_networks(context, filters, None, sorts, self).get_networks(context, filters, None, sorts,
limit, marker, page_reverse) limit, marker, page_reverse)
for net in nets:
self._extend_network_dict_provider(context, net)
return [self._fields(net, fields) for net in nets] return [self._fields(net, fields) for net in nets]