Add partial specs support in ML2 for gre/vxlan provider networks
ML2 provider networks partial specs let admins choose some provider network attributes and let neutron choose remaining attributes. This change provides the implementation for GRE/VXLAN provider networks. In practice, for GRE/VXLAN provider networks provider:segmentation_id choice can be delegated to neutron, in such case neutron will try to find a network in tenant network pools which respects provided provider attributes. DocImpact Related to blueprint provider-network-partial-specs Partial-Bug: #1330562 Change-Id: I720d7d04f6e3453145e888d9e4d9b5e381d0f7d4
This commit is contained in:
parent
4a2f3e5092
commit
7da4f601ee
@ -25,6 +25,7 @@ from neutron.db import model_base
|
|||||||
from neutron.openstack.common import log
|
from neutron.openstack.common import log
|
||||||
from neutron.plugins.common import constants as p_const
|
from neutron.plugins.common import constants as p_const
|
||||||
from neutron.plugins.ml2 import driver_api as api
|
from neutron.plugins.ml2 import driver_api as api
|
||||||
|
from neutron.plugins.ml2.drivers import helpers
|
||||||
from neutron.plugins.ml2.drivers import type_tunnel
|
from neutron.plugins.ml2.drivers import type_tunnel
|
||||||
|
|
||||||
LOG = log.getLogger(__name__)
|
LOG = log.getLogger(__name__)
|
||||||
@ -60,7 +61,10 @@ class GreEndpoints(model_base.BASEV2):
|
|||||||
return "<GreTunnelEndpoint(%s)>" % self.ip_address
|
return "<GreTunnelEndpoint(%s)>" % self.ip_address
|
||||||
|
|
||||||
|
|
||||||
class GreTypeDriver(type_tunnel.TunnelTypeDriver):
|
class GreTypeDriver(helpers.TypeDriverHelper, type_tunnel.TunnelTypeDriver):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super(GreTypeDriver, self).__init__(GreAllocation)
|
||||||
|
|
||||||
def get_type(self):
|
def get_type(self):
|
||||||
return p_const.TYPE_GRE
|
return p_const.TYPE_GRE
|
||||||
@ -75,39 +79,27 @@ class GreTypeDriver(type_tunnel.TunnelTypeDriver):
|
|||||||
self._sync_gre_allocations()
|
self._sync_gre_allocations()
|
||||||
|
|
||||||
def reserve_provider_segment(self, session, segment):
|
def reserve_provider_segment(self, session, segment):
|
||||||
segmentation_id = segment.get(api.SEGMENTATION_ID)
|
if self.is_partial_segment(segment):
|
||||||
with session.begin(subtransactions=True):
|
alloc = self.allocate_partially_specified_segment(session)
|
||||||
try:
|
if not alloc:
|
||||||
alloc = (session.query(GreAllocation).
|
raise exc.NoNetworkAvailable
|
||||||
filter_by(gre_id=segmentation_id).
|
else:
|
||||||
with_lockmode('update').
|
segmentation_id = segment.get(api.SEGMENTATION_ID)
|
||||||
one())
|
alloc = self.allocate_fully_specified_segment(
|
||||||
if alloc.allocated:
|
session, gre_id=segmentation_id)
|
||||||
raise exc.TunnelIdInUse(tunnel_id=segmentation_id)
|
if not alloc:
|
||||||
LOG.debug(_("Reserving specific gre tunnel %s from pool"),
|
raise exc.TunnelIdInUse(tunnel_id=segmentation_id)
|
||||||
segmentation_id)
|
return {api.NETWORK_TYPE: p_const.TYPE_GRE,
|
||||||
alloc.allocated = True
|
api.PHYSICAL_NETWORK: None,
|
||||||
except sa_exc.NoResultFound:
|
api.SEGMENTATION_ID: alloc.gre_id}
|
||||||
LOG.debug(_("Reserving specific gre tunnel %s outside pool"),
|
|
||||||
segmentation_id)
|
|
||||||
alloc = GreAllocation(gre_id=segmentation_id)
|
|
||||||
alloc.allocated = True
|
|
||||||
session.add(alloc)
|
|
||||||
return segment
|
|
||||||
|
|
||||||
def allocate_tenant_segment(self, session):
|
def allocate_tenant_segment(self, session):
|
||||||
with session.begin(subtransactions=True):
|
alloc = self.allocate_partially_specified_segment(session)
|
||||||
alloc = (session.query(GreAllocation).
|
if not alloc:
|
||||||
filter_by(allocated=False).
|
return
|
||||||
with_lockmode('update').
|
return {api.NETWORK_TYPE: p_const.TYPE_GRE,
|
||||||
first())
|
api.PHYSICAL_NETWORK: None,
|
||||||
if alloc:
|
api.SEGMENTATION_ID: alloc.gre_id}
|
||||||
LOG.debug(_("Allocating gre tunnel id %(gre_id)s"),
|
|
||||||
{'gre_id': alloc.gre_id})
|
|
||||||
alloc.allocated = True
|
|
||||||
return {api.NETWORK_TYPE: p_const.TYPE_GRE,
|
|
||||||
api.PHYSICAL_NETWORK: None,
|
|
||||||
api.SEGMENTATION_ID: alloc.gre_id}
|
|
||||||
|
|
||||||
def release_segment(self, session, segment):
|
def release_segment(self, session, segment):
|
||||||
gre_id = segment[api.SEGMENTATION_ID]
|
gre_id = segment[api.SEGMENTATION_ID]
|
||||||
|
@ -64,6 +64,9 @@ class TunnelTypeDriver(api.TypeDriver):
|
|||||||
LOG.info(_("%(type)s ID ranges: %(range)s"),
|
LOG.info(_("%(type)s ID ranges: %(range)s"),
|
||||||
{'type': tunnel_type, 'range': current_range})
|
{'type': tunnel_type, 'range': current_range})
|
||||||
|
|
||||||
|
def is_partial_segment(self, segment):
|
||||||
|
return segment.get(api.SEGMENTATION_ID) is None
|
||||||
|
|
||||||
def validate_provider_segment(self, segment):
|
def validate_provider_segment(self, segment):
|
||||||
physical_network = segment.get(api.PHYSICAL_NETWORK)
|
physical_network = segment.get(api.PHYSICAL_NETWORK)
|
||||||
if physical_network:
|
if physical_network:
|
||||||
@ -71,12 +74,6 @@ class TunnelTypeDriver(api.TypeDriver):
|
|||||||
"network") % segment.get(api.NETWORK_TYPE)
|
"network") % segment.get(api.NETWORK_TYPE)
|
||||||
raise exc.InvalidInput(error_message=msg)
|
raise exc.InvalidInput(error_message=msg)
|
||||||
|
|
||||||
segmentation_id = segment.get(api.SEGMENTATION_ID)
|
|
||||||
if not segmentation_id:
|
|
||||||
msg = _("segmentation_id required for %s provider "
|
|
||||||
"network") % segment.get(api.NETWORK_TYPE)
|
|
||||||
raise exc.InvalidInput(error_message=msg)
|
|
||||||
|
|
||||||
for key, value in segment.items():
|
for key, value in segment.items():
|
||||||
if value and key not in [api.NETWORK_TYPE,
|
if value and key not in [api.NETWORK_TYPE,
|
||||||
api.SEGMENTATION_ID]:
|
api.SEGMENTATION_ID]:
|
||||||
|
@ -25,6 +25,7 @@ from neutron.db import model_base
|
|||||||
from neutron.openstack.common import log
|
from neutron.openstack.common import log
|
||||||
from neutron.plugins.common import constants as p_const
|
from neutron.plugins.common import constants as p_const
|
||||||
from neutron.plugins.ml2 import driver_api as api
|
from neutron.plugins.ml2 import driver_api as api
|
||||||
|
from neutron.plugins.ml2.drivers import helpers
|
||||||
from neutron.plugins.ml2.drivers import type_tunnel
|
from neutron.plugins.ml2.drivers import type_tunnel
|
||||||
|
|
||||||
LOG = log.getLogger(__name__)
|
LOG = log.getLogger(__name__)
|
||||||
@ -68,7 +69,10 @@ class VxlanEndpoints(model_base.BASEV2):
|
|||||||
return "<VxlanTunnelEndpoint(%s)>" % self.ip_address
|
return "<VxlanTunnelEndpoint(%s)>" % self.ip_address
|
||||||
|
|
||||||
|
|
||||||
class VxlanTypeDriver(type_tunnel.TunnelTypeDriver):
|
class VxlanTypeDriver(helpers.TypeDriverHelper, type_tunnel.TunnelTypeDriver):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super(VxlanTypeDriver, self).__init__(VxlanAllocation)
|
||||||
|
|
||||||
def get_type(self):
|
def get_type(self):
|
||||||
return p_const.TYPE_VXLAN
|
return p_const.TYPE_VXLAN
|
||||||
@ -83,39 +87,27 @@ class VxlanTypeDriver(type_tunnel.TunnelTypeDriver):
|
|||||||
self._sync_vxlan_allocations()
|
self._sync_vxlan_allocations()
|
||||||
|
|
||||||
def reserve_provider_segment(self, session, segment):
|
def reserve_provider_segment(self, session, segment):
|
||||||
segmentation_id = segment.get(api.SEGMENTATION_ID)
|
if self.is_partial_segment(segment):
|
||||||
with session.begin(subtransactions=True):
|
alloc = self.allocate_partially_specified_segment(session)
|
||||||
try:
|
if not alloc:
|
||||||
alloc = (session.query(VxlanAllocation).
|
raise exc.NoNetworkAvailable
|
||||||
filter_by(vxlan_vni=segmentation_id).
|
else:
|
||||||
with_lockmode('update').
|
segmentation_id = segment.get(api.SEGMENTATION_ID)
|
||||||
one())
|
alloc = self.allocate_fully_specified_segment(
|
||||||
if alloc.allocated:
|
session, vxlan_vni=segmentation_id)
|
||||||
raise exc.TunnelIdInUse(tunnel_id=segmentation_id)
|
if not alloc:
|
||||||
LOG.debug(_("Reserving specific vxlan tunnel %s from pool"),
|
raise exc.TunnelIdInUse(tunnel_id=segmentation_id)
|
||||||
segmentation_id)
|
return {api.NETWORK_TYPE: p_const.TYPE_VXLAN,
|
||||||
alloc.allocated = True
|
api.PHYSICAL_NETWORK: None,
|
||||||
except sa_exc.NoResultFound:
|
api.SEGMENTATION_ID: alloc.vxlan_vni}
|
||||||
LOG.debug(_("Reserving specific vxlan tunnel %s outside pool"),
|
|
||||||
segmentation_id)
|
|
||||||
alloc = VxlanAllocation(vxlan_vni=segmentation_id)
|
|
||||||
alloc.allocated = True
|
|
||||||
session.add(alloc)
|
|
||||||
return segment
|
|
||||||
|
|
||||||
def allocate_tenant_segment(self, session):
|
def allocate_tenant_segment(self, session):
|
||||||
with session.begin(subtransactions=True):
|
alloc = self.allocate_partially_specified_segment(session)
|
||||||
alloc = (session.query(VxlanAllocation).
|
if not alloc:
|
||||||
filter_by(allocated=False).
|
return
|
||||||
with_lockmode('update').
|
return {api.NETWORK_TYPE: p_const.TYPE_VXLAN,
|
||||||
first())
|
api.PHYSICAL_NETWORK: None,
|
||||||
if alloc:
|
api.SEGMENTATION_ID: alloc.vxlan_vni}
|
||||||
LOG.debug(_("Allocating vxlan tunnel vni %(vxlan_vni)s"),
|
|
||||||
{'vxlan_vni': alloc.vxlan_vni})
|
|
||||||
alloc.allocated = True
|
|
||||||
return {api.NETWORK_TYPE: p_const.TYPE_VXLAN,
|
|
||||||
api.PHYSICAL_NETWORK: None,
|
|
||||||
api.SEGMENTATION_ID: alloc.vxlan_vni}
|
|
||||||
|
|
||||||
def release_segment(self, session, segment):
|
def release_segment(self, session, segment):
|
||||||
vxlan_vni = segment[api.SEGMENTATION_ID]
|
vxlan_vni = segment[api.SEGMENTATION_ID]
|
||||||
|
@ -51,8 +51,10 @@ class GreTypeTest(base.BaseTestCase):
|
|||||||
self.driver.validate_provider_segment(segment)
|
self.driver.validate_provider_segment(segment)
|
||||||
|
|
||||||
segment[api.PHYSICAL_NETWORK] = None
|
segment[api.PHYSICAL_NETWORK] = None
|
||||||
with testtools.ExpectedException(exc.InvalidInput):
|
self.driver.validate_provider_segment(segment)
|
||||||
self.driver.validate_provider_segment(segment)
|
|
||||||
|
segment[api.SEGMENTATION_ID] = 1
|
||||||
|
self.driver.validate_provider_segment(segment)
|
||||||
|
|
||||||
def test_sync_tunnel_allocations(self):
|
def test_sync_tunnel_allocations(self):
|
||||||
self.assertIsNone(
|
self.assertIsNone(
|
||||||
@ -108,9 +110,21 @@ class GreTypeTest(base.BaseTestCase):
|
|||||||
(TUN_MAX + 5 + 1))
|
(TUN_MAX + 5 + 1))
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_reserve_provider_segment(self):
|
def test_partial_segment_is_partial_segment(self):
|
||||||
segment = {api.NETWORK_TYPE: 'gre',
|
segment = {api.NETWORK_TYPE: 'gre',
|
||||||
api.PHYSICAL_NETWORK: 'None',
|
api.PHYSICAL_NETWORK: None,
|
||||||
|
api.SEGMENTATION_ID: None}
|
||||||
|
self.assertTrue(self.driver.is_partial_segment(segment))
|
||||||
|
|
||||||
|
def test_specific_segment_is_not_partial_segment(self):
|
||||||
|
segment = {api.NETWORK_TYPE: 'gre',
|
||||||
|
api.PHYSICAL_NETWORK: None,
|
||||||
|
api.SEGMENTATION_ID: 101}
|
||||||
|
self.assertFalse(self.driver.is_partial_segment(segment))
|
||||||
|
|
||||||
|
def test_reserve_provider_segment_full_specs(self):
|
||||||
|
segment = {api.NETWORK_TYPE: 'gre',
|
||||||
|
api.PHYSICAL_NETWORK: None,
|
||||||
api.SEGMENTATION_ID: 101}
|
api.SEGMENTATION_ID: 101}
|
||||||
observed = self.driver.reserve_provider_segment(self.session, segment)
|
observed = self.driver.reserve_provider_segment(self.session, segment)
|
||||||
alloc = self.driver.get_gre_allocation(self.session,
|
alloc = self.driver.get_gre_allocation(self.session,
|
||||||
@ -136,6 +150,41 @@ class GreTypeTest(base.BaseTestCase):
|
|||||||
observed[api.SEGMENTATION_ID])
|
observed[api.SEGMENTATION_ID])
|
||||||
self.assertIsNone(alloc)
|
self.assertIsNone(alloc)
|
||||||
|
|
||||||
|
def test_reserve_provider_segment(self):
|
||||||
|
tunnel_ids = set()
|
||||||
|
specs = {api.NETWORK_TYPE: 'gre',
|
||||||
|
api.PHYSICAL_NETWORK: 'None',
|
||||||
|
api.SEGMENTATION_ID: None}
|
||||||
|
|
||||||
|
for x in xrange(TUN_MIN, TUN_MAX + 1):
|
||||||
|
segment = self.driver.reserve_provider_segment(self.session,
|
||||||
|
specs)
|
||||||
|
self.assertEqual('gre', segment[api.NETWORK_TYPE])
|
||||||
|
self.assertThat(segment[api.SEGMENTATION_ID],
|
||||||
|
matchers.GreaterThan(TUN_MIN - 1))
|
||||||
|
self.assertThat(segment[api.SEGMENTATION_ID],
|
||||||
|
matchers.LessThan(TUN_MAX + 1))
|
||||||
|
tunnel_ids.add(segment[api.SEGMENTATION_ID])
|
||||||
|
|
||||||
|
with testtools.ExpectedException(exc.NoNetworkAvailable):
|
||||||
|
segment = self.driver.reserve_provider_segment(self.session,
|
||||||
|
specs)
|
||||||
|
|
||||||
|
segment = {api.NETWORK_TYPE: 'gre',
|
||||||
|
api.PHYSICAL_NETWORK: 'None',
|
||||||
|
api.SEGMENTATION_ID: tunnel_ids.pop()}
|
||||||
|
self.driver.release_segment(self.session, segment)
|
||||||
|
segment = self.driver.reserve_provider_segment(self.session, specs)
|
||||||
|
self.assertThat(segment[api.SEGMENTATION_ID],
|
||||||
|
matchers.GreaterThan(TUN_MIN - 1))
|
||||||
|
self.assertThat(segment[api.SEGMENTATION_ID],
|
||||||
|
matchers.LessThan(TUN_MAX + 1))
|
||||||
|
tunnel_ids.add(segment[api.SEGMENTATION_ID])
|
||||||
|
|
||||||
|
for tunnel_id in tunnel_ids:
|
||||||
|
segment[api.SEGMENTATION_ID] = tunnel_id
|
||||||
|
self.driver.release_segment(self.session, segment)
|
||||||
|
|
||||||
def test_allocate_tenant_segment(self):
|
def test_allocate_tenant_segment(self):
|
||||||
tunnel_ids = set()
|
tunnel_ids = set()
|
||||||
for x in moves.xrange(TUN_MIN, TUN_MAX + 1):
|
for x in moves.xrange(TUN_MIN, TUN_MAX + 1):
|
||||||
|
@ -65,8 +65,10 @@ class VxlanTypeTest(base.BaseTestCase):
|
|||||||
self.driver.validate_provider_segment(segment)
|
self.driver.validate_provider_segment(segment)
|
||||||
|
|
||||||
segment[api.PHYSICAL_NETWORK] = None
|
segment[api.PHYSICAL_NETWORK] = None
|
||||||
with testtools.ExpectedException(exc.InvalidInput):
|
self.driver.validate_provider_segment(segment)
|
||||||
self.driver.validate_provider_segment(segment)
|
|
||||||
|
segment[api.SEGMENTATION_ID] = 1
|
||||||
|
self.driver.validate_provider_segment(segment)
|
||||||
|
|
||||||
def test_sync_tunnel_allocations(self):
|
def test_sync_tunnel_allocations(self):
|
||||||
self.assertIsNone(
|
self.assertIsNone(
|
||||||
@ -116,9 +118,21 @@ class VxlanTypeTest(base.BaseTestCase):
|
|||||||
get_vxlan_allocation(self.session,
|
get_vxlan_allocation(self.session,
|
||||||
(TUN_MAX + 5 + 1)))
|
(TUN_MAX + 5 + 1)))
|
||||||
|
|
||||||
def test_reserve_provider_segment(self):
|
def test_partial_segment_is_partial_segment(self):
|
||||||
segment = {api.NETWORK_TYPE: 'vxlan',
|
segment = {api.NETWORK_TYPE: 'vxlan',
|
||||||
api.PHYSICAL_NETWORK: 'None',
|
api.PHYSICAL_NETWORK: None,
|
||||||
|
api.SEGMENTATION_ID: None}
|
||||||
|
self.assertTrue(self.driver.is_partial_segment(segment))
|
||||||
|
|
||||||
|
def test_specific_segment_is_not_partial_segment(self):
|
||||||
|
segment = {api.NETWORK_TYPE: 'vxlan',
|
||||||
|
api.PHYSICAL_NETWORK: None,
|
||||||
|
api.SEGMENTATION_ID: 101}
|
||||||
|
self.assertFalse(self.driver.is_partial_segment(segment))
|
||||||
|
|
||||||
|
def test_reserve_provider_segment_full_specs(self):
|
||||||
|
segment = {api.NETWORK_TYPE: 'vxlan',
|
||||||
|
api.PHYSICAL_NETWORK: None,
|
||||||
api.SEGMENTATION_ID: 101}
|
api.SEGMENTATION_ID: 101}
|
||||||
observed = self.driver.reserve_provider_segment(self.session, segment)
|
observed = self.driver.reserve_provider_segment(self.session, segment)
|
||||||
alloc = self.driver.get_vxlan_allocation(self.session,
|
alloc = self.driver.get_vxlan_allocation(self.session,
|
||||||
@ -144,6 +158,41 @@ class VxlanTypeTest(base.BaseTestCase):
|
|||||||
observed[api.SEGMENTATION_ID])
|
observed[api.SEGMENTATION_ID])
|
||||||
self.assertIsNone(alloc)
|
self.assertIsNone(alloc)
|
||||||
|
|
||||||
|
def test_reserve_provider_segment(self):
|
||||||
|
tunnel_ids = set()
|
||||||
|
specs = {api.NETWORK_TYPE: 'vxlan',
|
||||||
|
api.PHYSICAL_NETWORK: 'None',
|
||||||
|
api.SEGMENTATION_ID: None}
|
||||||
|
|
||||||
|
for x in xrange(TUN_MIN, TUN_MAX + 1):
|
||||||
|
segment = self.driver.reserve_provider_segment(self.session,
|
||||||
|
specs)
|
||||||
|
self.assertEqual('vxlan', segment[api.NETWORK_TYPE])
|
||||||
|
self.assertThat(segment[api.SEGMENTATION_ID],
|
||||||
|
matchers.GreaterThan(TUN_MIN - 1))
|
||||||
|
self.assertThat(segment[api.SEGMENTATION_ID],
|
||||||
|
matchers.LessThan(TUN_MAX + 1))
|
||||||
|
tunnel_ids.add(segment[api.SEGMENTATION_ID])
|
||||||
|
|
||||||
|
with testtools.ExpectedException(exc.NoNetworkAvailable):
|
||||||
|
segment = self.driver.reserve_provider_segment(self.session,
|
||||||
|
specs)
|
||||||
|
|
||||||
|
segment = {api.NETWORK_TYPE: 'vxlan',
|
||||||
|
api.PHYSICAL_NETWORK: 'None',
|
||||||
|
api.SEGMENTATION_ID: tunnel_ids.pop()}
|
||||||
|
self.driver.release_segment(self.session, segment)
|
||||||
|
segment = self.driver.reserve_provider_segment(self.session, specs)
|
||||||
|
self.assertThat(segment[api.SEGMENTATION_ID],
|
||||||
|
matchers.GreaterThan(TUN_MIN - 1))
|
||||||
|
self.assertThat(segment[api.SEGMENTATION_ID],
|
||||||
|
matchers.LessThan(TUN_MAX + 1))
|
||||||
|
tunnel_ids.add(segment[api.SEGMENTATION_ID])
|
||||||
|
|
||||||
|
for tunnel_id in tunnel_ids:
|
||||||
|
segment[api.SEGMENTATION_ID] = tunnel_id
|
||||||
|
self.driver.release_segment(self.session, segment)
|
||||||
|
|
||||||
def test_allocate_tenant_segment(self):
|
def test_allocate_tenant_segment(self):
|
||||||
tunnel_ids = set()
|
tunnel_ids = set()
|
||||||
for x in moves.xrange(TUN_MIN, TUN_MAX + 1):
|
for x in moves.xrange(TUN_MIN, TUN_MAX + 1):
|
||||||
|
Loading…
x
Reference in New Issue
Block a user