c9fd72a8e1
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
258 lines
10 KiB
Python
258 lines
10 KiB
Python
# Copyright (c) 2013 OpenStack Foundation
|
|
# All Rights Reserved.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License. You may obtain
|
|
# a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations
|
|
# under the License.
|
|
|
|
from six import moves
|
|
import testtools
|
|
from testtools import matchers
|
|
|
|
from neutron.common import exceptions as exc
|
|
import neutron.db.api as db
|
|
from neutron.plugins.ml2 import driver_api as api
|
|
from neutron.plugins.ml2.drivers import type_gre
|
|
from neutron.tests import base
|
|
|
|
TUNNEL_IP_ONE = "10.10.10.10"
|
|
TUNNEL_IP_TWO = "10.10.10.20"
|
|
TUN_MIN = 100
|
|
TUN_MAX = 109
|
|
TUNNEL_RANGES = [(TUN_MIN, TUN_MAX)]
|
|
UPDATED_TUNNEL_RANGES = [(TUN_MIN + 5, TUN_MAX + 5)]
|
|
|
|
|
|
class GreTypeTest(base.BaseTestCase):
|
|
|
|
def setUp(self):
|
|
super(GreTypeTest, self).setUp()
|
|
db.configure_db()
|
|
self.driver = type_gre.GreTypeDriver()
|
|
self.driver.gre_id_ranges = TUNNEL_RANGES
|
|
self.driver._sync_gre_allocations()
|
|
self.session = db.get_session()
|
|
self.addCleanup(db.clear_db)
|
|
|
|
def test_validate_provider_segment(self):
|
|
segment = {api.NETWORK_TYPE: 'gre',
|
|
api.PHYSICAL_NETWORK: 'phys_net',
|
|
api.SEGMENTATION_ID: None}
|
|
|
|
with testtools.ExpectedException(exc.InvalidInput):
|
|
self.driver.validate_provider_segment(segment)
|
|
|
|
segment[api.PHYSICAL_NETWORK] = None
|
|
self.driver.validate_provider_segment(segment)
|
|
|
|
segment[api.SEGMENTATION_ID] = 1
|
|
self.driver.validate_provider_segment(segment)
|
|
|
|
def test_sync_tunnel_allocations(self):
|
|
self.assertIsNone(
|
|
self.driver.get_gre_allocation(self.session,
|
|
(TUN_MIN - 1))
|
|
)
|
|
self.assertFalse(
|
|
self.driver.get_gre_allocation(self.session,
|
|
(TUN_MIN)).allocated
|
|
)
|
|
self.assertFalse(
|
|
self.driver.get_gre_allocation(self.session,
|
|
(TUN_MIN + 1)).allocated
|
|
)
|
|
self.assertFalse(
|
|
self.driver.get_gre_allocation(self.session,
|
|
(TUN_MAX - 1)).allocated
|
|
)
|
|
self.assertFalse(
|
|
self.driver.get_gre_allocation(self.session,
|
|
(TUN_MAX)).allocated
|
|
)
|
|
self.assertIsNone(
|
|
self.driver.get_gre_allocation(self.session,
|
|
(TUN_MAX + 1))
|
|
)
|
|
|
|
self.driver.gre_id_ranges = UPDATED_TUNNEL_RANGES
|
|
self.driver._sync_gre_allocations()
|
|
|
|
self.assertIsNone(
|
|
self.driver.get_gre_allocation(self.session,
|
|
(TUN_MIN + 5 - 1))
|
|
)
|
|
self.assertFalse(
|
|
self.driver.get_gre_allocation(self.session,
|
|
(TUN_MIN + 5)).allocated
|
|
)
|
|
self.assertFalse(
|
|
self.driver.get_gre_allocation(self.session,
|
|
(TUN_MIN + 5 + 1)).allocated
|
|
)
|
|
self.assertFalse(
|
|
self.driver.get_gre_allocation(self.session,
|
|
(TUN_MAX + 5 - 1)).allocated
|
|
)
|
|
self.assertFalse(
|
|
self.driver.get_gre_allocation(self.session,
|
|
(TUN_MAX + 5)).allocated
|
|
)
|
|
self.assertIsNone(
|
|
self.driver.get_gre_allocation(self.session,
|
|
(TUN_MAX + 5 + 1))
|
|
)
|
|
|
|
def test_partial_segment_is_partial_segment(self):
|
|
segment = {api.NETWORK_TYPE: 'gre',
|
|
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}
|
|
observed = self.driver.reserve_provider_segment(self.session, segment)
|
|
alloc = self.driver.get_gre_allocation(self.session,
|
|
observed[api.SEGMENTATION_ID])
|
|
self.assertTrue(alloc.allocated)
|
|
|
|
with testtools.ExpectedException(exc.TunnelIdInUse):
|
|
self.driver.reserve_provider_segment(self.session, segment)
|
|
|
|
self.driver.release_segment(self.session, segment)
|
|
alloc = self.driver.get_gre_allocation(self.session,
|
|
observed[api.SEGMENTATION_ID])
|
|
self.assertFalse(alloc.allocated)
|
|
|
|
segment[api.SEGMENTATION_ID] = 1000
|
|
observed = self.driver.reserve_provider_segment(self.session, segment)
|
|
alloc = self.driver.get_gre_allocation(self.session,
|
|
observed[api.SEGMENTATION_ID])
|
|
self.assertTrue(alloc.allocated)
|
|
|
|
self.driver.release_segment(self.session, segment)
|
|
alloc = self.driver.get_gre_allocation(self.session,
|
|
observed[api.SEGMENTATION_ID])
|
|
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):
|
|
tunnel_ids = set()
|
|
for x in moves.xrange(TUN_MIN, TUN_MAX + 1):
|
|
segment = self.driver.allocate_tenant_segment(self.session)
|
|
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])
|
|
|
|
segment = self.driver.allocate_tenant_segment(self.session)
|
|
self.assertIsNone(segment)
|
|
|
|
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.allocate_tenant_segment(self.session)
|
|
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_gre_endpoints(self):
|
|
tun_1 = self.driver.add_endpoint(TUNNEL_IP_ONE)
|
|
tun_2 = self.driver.add_endpoint(TUNNEL_IP_TWO)
|
|
self.assertEqual(TUNNEL_IP_ONE, tun_1.ip_address)
|
|
self.assertEqual(TUNNEL_IP_TWO, tun_2.ip_address)
|
|
|
|
# Get all the endpoints
|
|
endpoints = self.driver.get_endpoints()
|
|
for endpoint in endpoints:
|
|
self.assertIn(endpoint['ip_address'],
|
|
[TUNNEL_IP_ONE, TUNNEL_IP_TWO])
|
|
|
|
|
|
class GreTypeMultiRangeTest(base.BaseTestCase):
|
|
|
|
TUN_MIN0 = 100
|
|
TUN_MAX0 = 101
|
|
TUN_MIN1 = 200
|
|
TUN_MAX1 = 201
|
|
TUNNEL_MULTI_RANGES = [(TUN_MIN0, TUN_MAX0), (TUN_MIN1, TUN_MAX1)]
|
|
|
|
def setUp(self):
|
|
super(GreTypeMultiRangeTest, self).setUp()
|
|
db.configure_db()
|
|
self.driver = type_gre.GreTypeDriver()
|
|
self.driver.gre_id_ranges = self.TUNNEL_MULTI_RANGES
|
|
self.driver._sync_gre_allocations()
|
|
self.session = db.get_session()
|
|
self.addCleanup(db.clear_db)
|
|
|
|
def test_release_segment(self):
|
|
segments = [self.driver.allocate_tenant_segment(self.session)
|
|
for i in range(4)]
|
|
|
|
# Release them in random order. No special meaning.
|
|
for i in (0, 2, 1, 3):
|
|
self.driver.release_segment(self.session, segments[i])
|
|
|
|
for key in (self.TUN_MIN0, self.TUN_MAX0,
|
|
self.TUN_MIN1, self.TUN_MAX1):
|
|
alloc = self.driver.get_gre_allocation(self.session, key)
|
|
self.assertFalse(alloc.allocated)
|