Fix interface-attach removes existing interfaces from db
The following commit 394c693e35
changed
allocate_port_for_instance() to only return the ports that were created
rather than all of the ports on the instance which broke the attach-interface
code.
This patch fixes this issue by remove the sync decorators from:
allocate_for_instance, allocate_port_for_instance, and
deallocate_port_for_instance to just _build_instance_nw_info which
is called in all these cases instead.
Closes-bug: #1223859
Change-Id: I66eb0c0ab926e0a8d1e2c9cfe1f7fd579ea3aa27
This commit is contained in:
parent
686da61dd3
commit
1957339df3
@ -191,7 +191,6 @@ class API(base.Base):
|
|||||||
raise exception.PortLimitExceeded()
|
raise exception.PortLimitExceeded()
|
||||||
raise
|
raise
|
||||||
|
|
||||||
@refresh_cache
|
|
||||||
def allocate_for_instance(self, context, instance, **kwargs):
|
def allocate_for_instance(self, context, instance, **kwargs):
|
||||||
"""Allocate network resources for the instance.
|
"""Allocate network resources for the instance.
|
||||||
|
|
||||||
@ -416,14 +415,12 @@ class API(base.Base):
|
|||||||
LOG.exception(_("Failed to delete neutron port %(portid)s")
|
LOG.exception(_("Failed to delete neutron port %(portid)s")
|
||||||
% {'portid': port})
|
% {'portid': port})
|
||||||
|
|
||||||
@refresh_cache
|
|
||||||
def allocate_port_for_instance(self, context, instance, port_id,
|
def allocate_port_for_instance(self, context, instance, port_id,
|
||||||
network_id=None, requested_ip=None):
|
network_id=None, requested_ip=None):
|
||||||
"""Allocate a port for the instance."""
|
"""Allocate a port for the instance."""
|
||||||
return self.allocate_for_instance(context, instance,
|
return self.allocate_for_instance(context, instance,
|
||||||
requested_networks=[(network_id, requested_ip, port_id)])
|
requested_networks=[(network_id, requested_ip, port_id)])
|
||||||
|
|
||||||
@refresh_cache
|
|
||||||
def deallocate_port_for_instance(self, context, instance, port_id):
|
def deallocate_port_for_instance(self, context, instance, port_id):
|
||||||
"""Remove a specified port from the instance.
|
"""Remove a specified port from the instance.
|
||||||
|
|
||||||
@ -454,6 +451,7 @@ class API(base.Base):
|
|||||||
update_cells=False)
|
update_cells=False)
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
@refresh_cache
|
||||||
def _get_instance_nw_info(self, context, instance, networks=None):
|
def _get_instance_nw_info(self, context, instance, networks=None):
|
||||||
LOG.debug(_('get_instance_nw_info() for %s'),
|
LOG.debug(_('get_instance_nw_info() for %s'),
|
||||||
instance['display_name'])
|
instance['display_name'])
|
||||||
@ -969,11 +967,15 @@ class API(base.Base):
|
|||||||
return network, ovs_interfaceid
|
return network, ovs_interfaceid
|
||||||
|
|
||||||
def _build_network_info_model(self, context, instance, networks=None):
|
def _build_network_info_model(self, context, instance, networks=None):
|
||||||
|
# Note(arosen): on interface-attach networks only contains the
|
||||||
|
# network that the interface is being attached to.
|
||||||
|
|
||||||
search_opts = {'tenant_id': instance['project_id'],
|
search_opts = {'tenant_id': instance['project_id'],
|
||||||
'device_id': instance['uuid'], }
|
'device_id': instance['uuid'], }
|
||||||
client = neutronv2.get_client(context, admin=True)
|
client = neutronv2.get_client(context, admin=True)
|
||||||
data = client.list_ports(**search_opts)
|
data = client.list_ports(**search_opts)
|
||||||
ports = data.get('ports', [])
|
ports = data.get('ports', [])
|
||||||
|
nw_info = network_model.NetworkInfo()
|
||||||
if networks is None:
|
if networks is None:
|
||||||
# retrieve networks from info_cache to get correct nic order
|
# retrieve networks from info_cache to get correct nic order
|
||||||
network_cache = self.conductor_api.instance_get_by_uuid(
|
network_cache = self.conductor_api.instance_get_by_uuid(
|
||||||
@ -986,12 +988,31 @@ class API(base.Base):
|
|||||||
# ensure ports are in preferred network order, and filter out
|
# ensure ports are in preferred network order, and filter out
|
||||||
# those not attached to one of the provided list of networks
|
# those not attached to one of the provided list of networks
|
||||||
else:
|
else:
|
||||||
|
|
||||||
|
# Unfortunately, this is sometimes in unicode and sometimes not
|
||||||
|
if isinstance(instance['info_cache']['network_info'], unicode):
|
||||||
|
ifaces = jsonutils.loads(
|
||||||
|
instance['info_cache']['network_info'])
|
||||||
|
else:
|
||||||
|
ifaces = instance['info_cache']['network_info']
|
||||||
|
|
||||||
|
# Include existing interfaces so they are not removed from the db.
|
||||||
|
# Needed when interfaces are added to existing instances.
|
||||||
|
for iface in ifaces:
|
||||||
|
nw_info.append(network_model.VIF(
|
||||||
|
id=iface['id'],
|
||||||
|
address=iface['address'],
|
||||||
|
network=iface['network'],
|
||||||
|
type=iface['type'],
|
||||||
|
ovs_interfaceid=iface['ovs_interfaceid'],
|
||||||
|
devname=iface['devname']))
|
||||||
|
|
||||||
net_ids = [n['id'] for n in networks]
|
net_ids = [n['id'] for n in networks]
|
||||||
|
|
||||||
ports = [port for port in ports if port['network_id'] in net_ids]
|
ports = [port for port in ports if port['network_id'] in net_ids]
|
||||||
_ensure_requested_network_ordering(lambda x: x['network_id'],
|
_ensure_requested_network_ordering(lambda x: x['network_id'],
|
||||||
ports, net_ids)
|
ports, net_ids)
|
||||||
|
|
||||||
nw_info = network_model.NetworkInfo()
|
|
||||||
for port in ports:
|
for port in ports:
|
||||||
network_IPs = self._nw_info_get_ips(client, port)
|
network_IPs = self._nw_info_get_ips(client, port)
|
||||||
subnets = self._nw_info_get_subnets(context, port, network_IPs)
|
subnets = self._nw_info_get_subnets(context, port, network_IPs)
|
||||||
|
@ -164,6 +164,7 @@ class TestNeutronSecurityGroups(
|
|||||||
self._create_network()
|
self._create_network()
|
||||||
fake_instance = {'project_id': 'fake_tenant',
|
fake_instance = {'project_id': 'fake_tenant',
|
||||||
'availability_zone': 'zone_one',
|
'availability_zone': 'zone_one',
|
||||||
|
'info_cache': {'network_info': []},
|
||||||
'security_groups': [],
|
'security_groups': [],
|
||||||
'uuid': str(uuid.uuid4()),
|
'uuid': str(uuid.uuid4()),
|
||||||
'display_name': 'test_instance'}
|
'display_name': 'test_instance'}
|
||||||
|
@ -512,11 +512,64 @@ class TestNeutronv2(TestNeutronv2Base):
|
|||||||
admin=True).MultipleTimes().AndReturn(
|
admin=True).MultipleTimes().AndReturn(
|
||||||
self.moxed_client)
|
self.moxed_client)
|
||||||
self.mox.ReplayAll()
|
self.mox.ReplayAll()
|
||||||
|
self.instance['info_cache'] = {'network_info': []}
|
||||||
nw_inf = api.get_instance_nw_info(self.context,
|
nw_inf = api.get_instance_nw_info(self.context,
|
||||||
self.instance,
|
self.instance,
|
||||||
networks=self.nets1)
|
networks=self.nets1)
|
||||||
self._verify_nw_info(nw_inf, 0)
|
self._verify_nw_info(nw_inf, 0)
|
||||||
|
|
||||||
|
def test_get_instance_nw_info_with_nets_and_info_cache(self):
|
||||||
|
# This tests that adding an interface to an instance does not
|
||||||
|
# remove the first instance from the instance.
|
||||||
|
api = neutronapi.API()
|
||||||
|
self.mox.StubOutWithMock(api.db, 'instance_info_cache_update')
|
||||||
|
api.db.instance_info_cache_update(
|
||||||
|
mox.IgnoreArg(),
|
||||||
|
self.instance['uuid'], mox.IgnoreArg())
|
||||||
|
self.moxed_client.list_ports(
|
||||||
|
tenant_id=self.instance['project_id'],
|
||||||
|
device_id=self.instance['uuid']).AndReturn(
|
||||||
|
{'ports': self.port_data1})
|
||||||
|
port_data = self.port_data1
|
||||||
|
for ip in port_data[0]['fixed_ips']:
|
||||||
|
self.moxed_client.list_floatingips(
|
||||||
|
fixed_ip_address=ip['ip_address'],
|
||||||
|
port_id=port_data[0]['id']).AndReturn(
|
||||||
|
{'floatingips': self.float_data1})
|
||||||
|
self.moxed_client.list_subnets(
|
||||||
|
id=mox.SameElementsAs(['my_subid1'])).AndReturn(
|
||||||
|
{'subnets': self.subnet_data1})
|
||||||
|
self.moxed_client.list_ports(
|
||||||
|
network_id='my_netid1',
|
||||||
|
device_owner='network:dhcp').AndReturn(
|
||||||
|
{'ports': self.dhcp_port_data1})
|
||||||
|
neutronv2.get_client(mox.IgnoreArg(),
|
||||||
|
admin=True).MultipleTimes().AndReturn(
|
||||||
|
self.moxed_client)
|
||||||
|
self.mox.ReplayAll()
|
||||||
|
network_model = model.Network(id='network_id',
|
||||||
|
bridge='br-int',
|
||||||
|
injected='injected',
|
||||||
|
label='fake_network',
|
||||||
|
tenant_name='fake_tenant')
|
||||||
|
|
||||||
|
self.instance['info_cache'] = {
|
||||||
|
'network_info': [{'id': 'port_id',
|
||||||
|
'address': 'mac_address',
|
||||||
|
'network': network_model,
|
||||||
|
'type': 'ovs',
|
||||||
|
'ovs_interfaceid': 'ovs_interfaceid',
|
||||||
|
'devname': 'devname'}]}
|
||||||
|
nw_inf = api.get_instance_nw_info(self.context,
|
||||||
|
self.instance,
|
||||||
|
networks=self.nets1)
|
||||||
|
self.assertEqual(2, len(nw_inf))
|
||||||
|
for k, v in self.instance['info_cache']['network_info'][0].iteritems():
|
||||||
|
self.assertEqual(nw_inf[0][k], v)
|
||||||
|
# remove first inf and verify that the second interface is correct
|
||||||
|
del nw_inf[0]
|
||||||
|
self._verify_nw_info(nw_inf, 0)
|
||||||
|
|
||||||
def test_get_instance_nw_info_without_subnet(self):
|
def test_get_instance_nw_info_without_subnet(self):
|
||||||
# Test get instance_nw_info for a port without subnet.
|
# Test get instance_nw_info for a port without subnet.
|
||||||
api = neutronapi.API()
|
api = neutronapi.API()
|
||||||
@ -1554,7 +1607,8 @@ class TestNeutronv2(TestNeutronv2Base):
|
|||||||
|
|
||||||
def test_build_network_info_model(self):
|
def test_build_network_info_model(self):
|
||||||
api = neutronapi.API()
|
api = neutronapi.API()
|
||||||
fake_inst = {'project_id': 'fake', 'uuid': 'uuid'}
|
fake_inst = {'project_id': 'fake', 'uuid': 'uuid',
|
||||||
|
'info_cache': {'network_info': []}}
|
||||||
fake_ports = [
|
fake_ports = [
|
||||||
{'id': 'port0',
|
{'id': 'port0',
|
||||||
'network_id': 'net-id',
|
'network_id': 'net-id',
|
||||||
|
Loading…
Reference in New Issue
Block a user