Add extra router attributes for ECMP and BFD
* enable_default_route_ecmp * enable_default_route_bfd Partial-Bug: #2002687 Change-Id: I3fcd0458d20f20ce40378f90f073f37c41400865
This commit is contained in:
parent
656897f32e
commit
89702218db
36
neutron/conf/db/l3_extra_gws_db.py
Normal file
36
neutron/conf/db/l3_extra_gws_db.py
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
# Copyright (c) 2023 Canonical Ltd.
|
||||||
|
#
|
||||||
|
# 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 oslo_config import cfg
|
||||||
|
|
||||||
|
from neutron._i18n import _
|
||||||
|
|
||||||
|
|
||||||
|
L3_EXTRA_GWS_OPTS = [
|
||||||
|
cfg.BoolOpt('enable_default_route_ecmp',
|
||||||
|
default=False,
|
||||||
|
help=_("Define the default value for "
|
||||||
|
"enable_default_route_ecmp if not specified on the "
|
||||||
|
"router.")),
|
||||||
|
cfg.BoolOpt('enable_default_route_bfd',
|
||||||
|
default=False,
|
||||||
|
help=_("Define the default value for "
|
||||||
|
"enable_default_route_bfd if not specified on the "
|
||||||
|
"router.")),
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
def register_db_l3_extragws_opts(conf=cfg.CONF):
|
||||||
|
conf.register_opts(L3_EXTRA_GWS_OPTS)
|
@ -127,6 +127,22 @@ rules = [
|
|||||||
deprecated_reason=DEPRECATED_REASON,
|
deprecated_reason=DEPRECATED_REASON,
|
||||||
deprecated_since=versionutils.deprecated.WALLABY)
|
deprecated_since=versionutils.deprecated.WALLABY)
|
||||||
),
|
),
|
||||||
|
policy.DocumentedRuleDefault(
|
||||||
|
name='create_router:enable_default_route_bfd',
|
||||||
|
check_str=base.ADMIN,
|
||||||
|
scope_types=['project'],
|
||||||
|
description=('Specify ``enable_default_route_bfd`` attribute when'
|
||||||
|
' creating a router'),
|
||||||
|
operations=ACTION_POST,
|
||||||
|
),
|
||||||
|
policy.DocumentedRuleDefault(
|
||||||
|
name='create_router:enable_default_route_ecmp',
|
||||||
|
check_str=base.ADMIN,
|
||||||
|
scope_types=['project'],
|
||||||
|
description=('Specify ``enable_default_route_ecmp`` attribute when'
|
||||||
|
' creating a router'),
|
||||||
|
operations=ACTION_POST,
|
||||||
|
),
|
||||||
|
|
||||||
policy.DocumentedRuleDefault(
|
policy.DocumentedRuleDefault(
|
||||||
name='get_router',
|
name='get_router',
|
||||||
@ -252,6 +268,22 @@ rules = [
|
|||||||
deprecated_reason=DEPRECATED_REASON,
|
deprecated_reason=DEPRECATED_REASON,
|
||||||
deprecated_since=versionutils.deprecated.WALLABY)
|
deprecated_since=versionutils.deprecated.WALLABY)
|
||||||
),
|
),
|
||||||
|
policy.DocumentedRuleDefault(
|
||||||
|
name='update_router:enable_default_route_bfd',
|
||||||
|
check_str=base.ADMIN,
|
||||||
|
scope_types=['project'],
|
||||||
|
description=('Specify ``enable_default_route_bfd`` attribute when '
|
||||||
|
'updating a router'),
|
||||||
|
operations=ACTION_POST,
|
||||||
|
),
|
||||||
|
policy.DocumentedRuleDefault(
|
||||||
|
name='update_router:enable_default_route_ecmp',
|
||||||
|
check_str=base.ADMIN,
|
||||||
|
scope_types=['project'],
|
||||||
|
description=('Specify ``enable_default_route_ecmp`` attribute when '
|
||||||
|
'updating a router'),
|
||||||
|
operations=ACTION_POST,
|
||||||
|
),
|
||||||
|
|
||||||
policy.DocumentedRuleDefault(
|
policy.DocumentedRuleDefault(
|
||||||
name='delete_router',
|
name='delete_router',
|
||||||
|
@ -18,9 +18,13 @@ from neutron_lib.db import resource_extend
|
|||||||
from oslo_config import cfg
|
from oslo_config import cfg
|
||||||
|
|
||||||
from neutron._i18n import _
|
from neutron._i18n import _
|
||||||
|
from neutron.conf.db import l3_extra_gws_db
|
||||||
from neutron.db.models import l3_attrs
|
from neutron.db.models import l3_attrs
|
||||||
|
|
||||||
|
|
||||||
|
l3_extra_gws_db.register_db_l3_extragws_opts()
|
||||||
|
|
||||||
|
|
||||||
def get_attr_info():
|
def get_attr_info():
|
||||||
"""Returns api visible attr names and their default values."""
|
"""Returns api visible attr names and their default values."""
|
||||||
return {'distributed': {'default': cfg.CONF.router_distributed},
|
return {'distributed': {'default': cfg.CONF.router_distributed},
|
||||||
@ -29,7 +33,11 @@ def get_attr_info():
|
|||||||
'availability_zone_hints': {
|
'availability_zone_hints': {
|
||||||
'default': '[]',
|
'default': '[]',
|
||||||
'transform_to_db': az_validator.convert_az_list_to_string,
|
'transform_to_db': az_validator.convert_az_list_to_string,
|
||||||
'transform_from_db': az_validator.convert_az_string_to_list}
|
'transform_from_db': az_validator.convert_az_string_to_list},
|
||||||
|
'enable_default_route_ecmp': {
|
||||||
|
'default': cfg.CONF.enable_default_route_ecmp},
|
||||||
|
'enable_default_route_bfd': {
|
||||||
|
'default': cfg.CONF.enable_default_route_bfd},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -22,6 +22,8 @@ from neutron.db import l3_gwmode_db
|
|||||||
from neutron.objects import ports as port_obj
|
from neutron.objects import ports as port_obj
|
||||||
from neutron.objects import router as l3_obj
|
from neutron.objects import router as l3_obj
|
||||||
from neutron_lib.api.definitions import l3 as l3_apidef
|
from neutron_lib.api.definitions import l3 as l3_apidef
|
||||||
|
from neutron_lib.api.definitions import l3_enable_default_route_bfd
|
||||||
|
from neutron_lib.api.definitions import l3_enable_default_route_ecmp
|
||||||
from neutron_lib.api.definitions import l3_ext_gw_multihoming
|
from neutron_lib.api.definitions import l3_ext_gw_multihoming
|
||||||
from neutron_lib.api import extensions
|
from neutron_lib.api import extensions
|
||||||
from neutron_lib.callbacks import events
|
from neutron_lib.callbacks import events
|
||||||
@ -75,6 +77,16 @@ class ExtraGatewaysDbOnlyMixin(l3_gwmode_db.L3_NAT_dbonly_mixin):
|
|||||||
trigger, payload):
|
trigger, payload):
|
||||||
self._remove_all_gateways(payload.context, payload.resource_id)
|
self._remove_all_gateways(payload.context, payload.resource_id)
|
||||||
|
|
||||||
|
@registry.receives(resources.ROUTER, [events.PRECOMMIT_CREATE])
|
||||||
|
def _process_bfd_ecmp_request(self, resource, event, trigger, payload):
|
||||||
|
router = payload.latest_state
|
||||||
|
router_db = payload.metadata['router_db']
|
||||||
|
for attr in (l3_enable_default_route_ecmp.ENABLE_DEFAULT_ROUTE_ECMP,
|
||||||
|
l3_enable_default_route_bfd.ENABLE_DEFAULT_ROUTE_BFD):
|
||||||
|
value = router.get(attr)
|
||||||
|
if value is not None:
|
||||||
|
self.set_extra_attr_value(router_db, attr, value)
|
||||||
|
|
||||||
def _add_external_gateways(
|
def _add_external_gateways(
|
||||||
self, context, router_id, gw_info_list, payload):
|
self, context, router_id, gw_info_list, payload):
|
||||||
"""Add external gateways to a router."""
|
"""Add external gateways to a router."""
|
||||||
|
@ -0,0 +1,39 @@
|
|||||||
|
# Copyright 2023 OpenStack Foundation
|
||||||
|
#
|
||||||
|
# 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 alembic import op
|
||||||
|
import sqlalchemy as sa
|
||||||
|
from sqlalchemy import sql
|
||||||
|
|
||||||
|
|
||||||
|
"""Add ECMP and BFD router-level policy attributes
|
||||||
|
Revision ID: 89c58a70ceba
|
||||||
|
Revises: c33da356b165
|
||||||
|
Create Date: 2023-02-22 21:08:33.593101
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision = '89c58a70ceba'
|
||||||
|
down_revision = 'c33da356b165'
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade():
|
||||||
|
op.add_column('router_extra_attributes',
|
||||||
|
sa.Column('enable_default_route_ecmp', sa.Boolean(),
|
||||||
|
server_default=sql.false(), nullable=False))
|
||||||
|
op.add_column('router_extra_attributes',
|
||||||
|
sa.Column('enable_default_route_bfd', sa.Boolean(),
|
||||||
|
server_default=sql.false(), nullable=False))
|
@ -1 +1 @@
|
|||||||
c33da356b165
|
89c58a70ceba
|
||||||
|
@ -40,6 +40,12 @@ class RouterExtraAttributes(model_base.BASEV2):
|
|||||||
ha_vr_id = sa.Column(sa.Integer())
|
ha_vr_id = sa.Column(sa.Integer())
|
||||||
# Availability Zone support
|
# Availability Zone support
|
||||||
availability_zone_hints = sa.Column(sa.String(255))
|
availability_zone_hints = sa.Column(sa.String(255))
|
||||||
|
enable_default_route_ecmp = sa.Column(sa.Boolean, default=False,
|
||||||
|
server_default=sa.sql.false(),
|
||||||
|
nullable=False)
|
||||||
|
enable_default_route_bfd = sa.Column(sa.Boolean, default=False,
|
||||||
|
server_default=sa.sql.false(),
|
||||||
|
nullable=False)
|
||||||
|
|
||||||
router = orm.relationship(
|
router = orm.relationship(
|
||||||
'Router', load_on_pending=True,
|
'Router', load_on_pending=True,
|
||||||
|
@ -71,7 +71,8 @@ class RouterRoute(base.NeutronDbObject):
|
|||||||
@base.NeutronObjectRegistry.register
|
@base.NeutronObjectRegistry.register
|
||||||
class RouterExtraAttributes(base.NeutronDbObject):
|
class RouterExtraAttributes(base.NeutronDbObject):
|
||||||
# Version 1.0: Initial version
|
# Version 1.0: Initial version
|
||||||
VERSION = '1.0'
|
# Version 1.1: Added ECMP and BFD attributes
|
||||||
|
VERSION = '1.1'
|
||||||
|
|
||||||
db_model = l3_attrs.RouterExtraAttributes
|
db_model = l3_attrs.RouterExtraAttributes
|
||||||
|
|
||||||
@ -81,7 +82,10 @@ class RouterExtraAttributes(base.NeutronDbObject):
|
|||||||
'service_router': obj_fields.BooleanField(default=False),
|
'service_router': obj_fields.BooleanField(default=False),
|
||||||
'ha': obj_fields.BooleanField(default=False),
|
'ha': obj_fields.BooleanField(default=False),
|
||||||
'ha_vr_id': obj_fields.IntegerField(nullable=True),
|
'ha_vr_id': obj_fields.IntegerField(nullable=True),
|
||||||
'availability_zone_hints': obj_fields.ListOfStringsField(nullable=True)
|
'availability_zone_hints': obj_fields.ListOfStringsField(
|
||||||
|
nullable=True),
|
||||||
|
'enable_default_route_bfd': obj_fields.BooleanField(default=False),
|
||||||
|
'enable_default_route_ecmp': obj_fields.BooleanField(default=False),
|
||||||
}
|
}
|
||||||
|
|
||||||
primary_keys = ['router_id']
|
primary_keys = ['router_id']
|
||||||
@ -130,6 +134,12 @@ class RouterExtraAttributes(base.NeutronDbObject):
|
|||||||
|
|
||||||
return list(query)
|
return list(query)
|
||||||
|
|
||||||
|
def obj_make_compatible(self, primitive, target_version):
|
||||||
|
_target_version = versionutils.convert_version_to_tuple(target_version)
|
||||||
|
if _target_version < (1, 1):
|
||||||
|
primitive.pop('enable_default_route_bfd', None)
|
||||||
|
primitive.pop('enable_default_route_ecmp', None)
|
||||||
|
|
||||||
|
|
||||||
@base.NeutronObjectRegistry.register
|
@base.NeutronObjectRegistry.register
|
||||||
class RouterPort(base.NeutronDbObject):
|
class RouterPort(base.NeutronDbObject):
|
||||||
|
@ -35,6 +35,7 @@ import neutron.conf.db.dvr_mac_db
|
|||||||
import neutron.conf.db.extraroute_db
|
import neutron.conf.db.extraroute_db
|
||||||
import neutron.conf.db.l3_agentschedulers_db
|
import neutron.conf.db.l3_agentschedulers_db
|
||||||
import neutron.conf.db.l3_dvr_db
|
import neutron.conf.db.l3_dvr_db
|
||||||
|
import neutron.conf.db.l3_extra_gws_db
|
||||||
import neutron.conf.db.l3_gwmode_db
|
import neutron.conf.db.l3_gwmode_db
|
||||||
import neutron.conf.db.l3_hamode_db
|
import neutron.conf.db.l3_hamode_db
|
||||||
import neutron.conf.experimental
|
import neutron.conf.experimental
|
||||||
@ -168,7 +169,8 @@ def list_db_opts():
|
|||||||
neutron.conf.db.dvr_mac_db.DVR_MAC_ADDRESS_OPTS,
|
neutron.conf.db.dvr_mac_db.DVR_MAC_ADDRESS_OPTS,
|
||||||
neutron.conf.db.l3_dvr_db.ROUTER_DISTRIBUTED_OPTS,
|
neutron.conf.db.l3_dvr_db.ROUTER_DISTRIBUTED_OPTS,
|
||||||
neutron.conf.db.l3_agentschedulers_db.L3_AGENTS_SCHEDULER_OPTS,
|
neutron.conf.db.l3_agentschedulers_db.L3_AGENTS_SCHEDULER_OPTS,
|
||||||
neutron.conf.db.l3_hamode_db.L3_HA_OPTS)
|
neutron.conf.db.l3_hamode_db.L3_HA_OPTS,
|
||||||
|
neutron.conf.db.l3_extra_gws_db.L3_EXTRA_GWS_OPTS)
|
||||||
),
|
),
|
||||||
('database',
|
('database',
|
||||||
neutron.db.migration.cli.get_engine_config())
|
neutron.db.migration.cli.get_engine_config())
|
||||||
|
@ -114,6 +114,30 @@ class SystemAdminTests(RouterAPITestCase):
|
|||||||
'create_router:external_gateway_info:external_fixed_ips',
|
'create_router:external_gateway_info:external_fixed_ips',
|
||||||
self.alt_target)
|
self.alt_target)
|
||||||
|
|
||||||
|
def test_create_router_enable_default_route_bfd(self):
|
||||||
|
self.assertRaises(
|
||||||
|
base_policy.InvalidScope,
|
||||||
|
policy.enforce,
|
||||||
|
self.context, 'create_router:enable_default_route_bfd',
|
||||||
|
self.target)
|
||||||
|
self.assertRaises(
|
||||||
|
base_policy.InvalidScope,
|
||||||
|
policy.enforce,
|
||||||
|
self.context, 'create_router:enable_default_route_bfd',
|
||||||
|
self.alt_target)
|
||||||
|
|
||||||
|
def test_create_router_enable_default_route_ecmp(self):
|
||||||
|
self.assertRaises(
|
||||||
|
base_policy.InvalidScope,
|
||||||
|
policy.enforce,
|
||||||
|
self.context, 'create_router:enable_default_route_ecmp',
|
||||||
|
self.target)
|
||||||
|
self.assertRaises(
|
||||||
|
base_policy.InvalidScope,
|
||||||
|
policy.enforce,
|
||||||
|
self.context, 'create_router:enable_default_route_ecmp',
|
||||||
|
self.alt_target)
|
||||||
|
|
||||||
def test_get_router(self):
|
def test_get_router(self):
|
||||||
self.assertRaises(
|
self.assertRaises(
|
||||||
base_policy.InvalidScope,
|
base_policy.InvalidScope,
|
||||||
@ -224,6 +248,30 @@ class SystemAdminTests(RouterAPITestCase):
|
|||||||
'update_router:external_gateway_info:external_fixed_ips',
|
'update_router:external_gateway_info:external_fixed_ips',
|
||||||
self.alt_target)
|
self.alt_target)
|
||||||
|
|
||||||
|
def test_update_router_enable_default_route_bfd(self):
|
||||||
|
self.assertRaises(
|
||||||
|
base_policy.InvalidScope,
|
||||||
|
policy.enforce,
|
||||||
|
self.context, 'update_router:enable_default_route_bfd',
|
||||||
|
self.target)
|
||||||
|
self.assertRaises(
|
||||||
|
base_policy.InvalidScope,
|
||||||
|
policy.enforce,
|
||||||
|
self.context, 'update_router:enable_default_route_bfd',
|
||||||
|
self.alt_target)
|
||||||
|
|
||||||
|
def test_update_router_enable_default_route_ecmp(self):
|
||||||
|
self.assertRaises(
|
||||||
|
base_policy.InvalidScope,
|
||||||
|
policy.enforce,
|
||||||
|
self.context, 'update_router:enable_default_route_ecmp',
|
||||||
|
self.target)
|
||||||
|
self.assertRaises(
|
||||||
|
base_policy.InvalidScope,
|
||||||
|
policy.enforce,
|
||||||
|
self.context, 'update_router:enable_default_route_ecmp',
|
||||||
|
self.alt_target)
|
||||||
|
|
||||||
def test_delete_router(self):
|
def test_delete_router(self):
|
||||||
self.assertRaises(
|
self.assertRaises(
|
||||||
base_policy.InvalidScope,
|
base_policy.InvalidScope,
|
||||||
@ -337,6 +385,30 @@ class AdminTests(RouterAPITestCase):
|
|||||||
'create_router:external_gateway_info:external_fixed_ips',
|
'create_router:external_gateway_info:external_fixed_ips',
|
||||||
self.alt_target))
|
self.alt_target))
|
||||||
|
|
||||||
|
def test_update_router_enable_default_route_bfd(self):
|
||||||
|
self.assertTrue(
|
||||||
|
policy.enforce(
|
||||||
|
self.context,
|
||||||
|
'update_router:enable_default_route_bfd',
|
||||||
|
self.target))
|
||||||
|
self.assertTrue(
|
||||||
|
policy.enforce(
|
||||||
|
self.context,
|
||||||
|
'update_router:enable_default_route_bfd',
|
||||||
|
self.alt_target))
|
||||||
|
|
||||||
|
def test_update_router_enable_default_route_ecmp(self):
|
||||||
|
self.assertTrue(
|
||||||
|
policy.enforce(
|
||||||
|
self.context,
|
||||||
|
'update_router:enable_default_route_ecmp',
|
||||||
|
self.target))
|
||||||
|
self.assertTrue(
|
||||||
|
policy.enforce(
|
||||||
|
self.context,
|
||||||
|
'update_router:enable_default_route_ecmp',
|
||||||
|
self.alt_target))
|
||||||
|
|
||||||
def test_get_router(self):
|
def test_get_router(self):
|
||||||
self.assertTrue(
|
self.assertTrue(
|
||||||
policy.enforce(self.context, 'get_router', self.target))
|
policy.enforce(self.context, 'get_router', self.target))
|
||||||
@ -524,6 +596,34 @@ class ProjectMemberTests(AdminTests):
|
|||||||
'create_router:external_gateway_info:external_fixed_ips',
|
'create_router:external_gateway_info:external_fixed_ips',
|
||||||
self.alt_target)
|
self.alt_target)
|
||||||
|
|
||||||
|
def test_update_router_enable_default_route_bfd(self):
|
||||||
|
self.assertRaises(
|
||||||
|
base_policy.PolicyNotAuthorized,
|
||||||
|
policy.enforce,
|
||||||
|
self.context,
|
||||||
|
'update_router:enable_default_route_bfd',
|
||||||
|
self.target)
|
||||||
|
self.assertRaises(
|
||||||
|
base_policy.PolicyNotAuthorized,
|
||||||
|
policy.enforce,
|
||||||
|
self.context,
|
||||||
|
'update_router:enable_default_route_bfd',
|
||||||
|
self.alt_target)
|
||||||
|
|
||||||
|
def test_update_router_enable_default_route_ecmp(self):
|
||||||
|
self.assertRaises(
|
||||||
|
base_policy.PolicyNotAuthorized,
|
||||||
|
policy.enforce,
|
||||||
|
self.context,
|
||||||
|
'update_router:enable_default_route_ecmp',
|
||||||
|
self.target)
|
||||||
|
self.assertRaises(
|
||||||
|
base_policy.PolicyNotAuthorized,
|
||||||
|
policy.enforce,
|
||||||
|
self.context,
|
||||||
|
'update_router:enable_default_route_ecmp',
|
||||||
|
self.alt_target)
|
||||||
|
|
||||||
def test_get_router(self):
|
def test_get_router(self):
|
||||||
self.assertTrue(
|
self.assertTrue(
|
||||||
policy.enforce(self.context, 'get_router', self.target))
|
policy.enforce(self.context, 'get_router', self.target))
|
||||||
|
@ -657,9 +657,17 @@ class ExtraAttributesMixinTestCase(testlib_api.SqlTestCase):
|
|||||||
self.mixin.set_extra_attr_value(self.router,
|
self.mixin.set_extra_attr_value(self.router,
|
||||||
'availability_zone_hints',
|
'availability_zone_hints',
|
||||||
['x', 'y', 'z'])
|
['x', 'y', 'z'])
|
||||||
|
self.mixin.set_extra_attr_value(self.router,
|
||||||
|
'enable_default_route_ecmp',
|
||||||
|
True)
|
||||||
|
self.mixin.set_extra_attr_value(self.router,
|
||||||
|
'enable_default_route_bfd',
|
||||||
|
True)
|
||||||
expected = self._get_default_api_values()
|
expected = self._get_default_api_values()
|
||||||
expected.update({'ha_vr_id': 99,
|
expected.update({'ha_vr_id': 99,
|
||||||
'availability_zone_hints': ['x', 'y', 'z']})
|
'availability_zone_hints': ['x', 'y', 'z'],
|
||||||
|
'enable_default_route_ecmp': True,
|
||||||
|
'enable_default_route_bfd': True})
|
||||||
rdict = {}
|
rdict = {}
|
||||||
self.mixin._extend_extra_router_dict(rdict, self.router)
|
self.mixin._extend_extra_router_dict(rdict, self.router)
|
||||||
self.assertEqual(expected, rdict)
|
self.assertEqual(expected, rdict)
|
||||||
@ -667,7 +675,15 @@ class ExtraAttributesMixinTestCase(testlib_api.SqlTestCase):
|
|||||||
self.mixin.set_extra_attr_value(self.router,
|
self.mixin.set_extra_attr_value(self.router,
|
||||||
'availability_zone_hints',
|
'availability_zone_hints',
|
||||||
['z', 'y', 'z'])
|
['z', 'y', 'z'])
|
||||||
|
self.mixin.set_extra_attr_value(self.router,
|
||||||
|
'enable_default_route_ecmp',
|
||||||
|
False)
|
||||||
|
self.mixin.set_extra_attr_value(self.router,
|
||||||
|
'enable_default_route_bfd',
|
||||||
|
False)
|
||||||
expected['availability_zone_hints'] = ['z', 'y', 'z']
|
expected['availability_zone_hints'] = ['z', 'y', 'z']
|
||||||
|
expected['enable_default_route_ecmp'] = False
|
||||||
|
expected['enable_default_route_bfd'] = False
|
||||||
self.mixin._extend_extra_router_dict(rdict, self.router)
|
self.mixin._extend_extra_router_dict(rdict, self.router)
|
||||||
self.assertEqual(expected, rdict)
|
self.assertEqual(expected, rdict)
|
||||||
|
|
||||||
|
@ -106,7 +106,7 @@ object_data = {
|
|||||||
'ResourceDelta': '1.0-a980b37e0a52618b5af8db29af18be76',
|
'ResourceDelta': '1.0-a980b37e0a52618b5af8db29af18be76',
|
||||||
'Route': '1.0-a9883a63b416126f9e345523ec09483b',
|
'Route': '1.0-a9883a63b416126f9e345523ec09483b',
|
||||||
'Router': '1.1-614fa16cc99c60e4fc19ac1b31a52291',
|
'Router': '1.1-614fa16cc99c60e4fc19ac1b31a52291',
|
||||||
'RouterExtraAttributes': '1.0-ef8d61ae2864f0ec9af0ab7939cab318',
|
'RouterExtraAttributes': '1.1-19c45c32098d2aae8e1a22d18944a954',
|
||||||
'RouterL3AgentBinding': '1.0-c5ba6c95e3a4c1236a55f490cd67da82',
|
'RouterL3AgentBinding': '1.0-c5ba6c95e3a4c1236a55f490cd67da82',
|
||||||
'RouterNDPProxyState': '1.0-4042e475bf173d1d8d17adb962eae1b2',
|
'RouterNDPProxyState': '1.0-4042e475bf173d1d8d17adb962eae1b2',
|
||||||
'RouterPort': '1.0-c8c8f499bcdd59186fcd83f323106908',
|
'RouterPort': '1.0-c8c8f499bcdd59186fcd83f323106908',
|
||||||
|
@ -0,0 +1,8 @@
|
|||||||
|
---
|
||||||
|
features:
|
||||||
|
- |
|
||||||
|
Add ``enable_default_route_bfd`` and ``enable_default_route_ecmp``
|
||||||
|
configuration options which control default behavior for enabling BFD and
|
||||||
|
ECMP on default routes for newly created routers. Both configuration
|
||||||
|
options have a default value of 'False' and are only supported with the
|
||||||
|
OVN driver.
|
Loading…
Reference in New Issue
Block a user