Make the MTU attribute not nullable
This patch sets the MTU attribute to non-nullable, the code that get the MTU and update the network in some methods can be removed. If the MTU is empty before the pike version, it is set to the default value of 1500. Change-Id: Id4d738dde7fa4b7caccabad0aac542b82b4d7af1 Closes-Bug: #1842261
This commit is contained in:
parent
f5bdaec2c7
commit
68625686a4
@ -22,6 +22,7 @@ from oslo_upgradecheck import upgradecheck
|
|||||||
from neutron._i18n import _
|
from neutron._i18n import _
|
||||||
from neutron.cmd.upgrade_checks import base
|
from neutron.cmd.upgrade_checks import base
|
||||||
from neutron.db.models import agent as agent_model
|
from neutron.db.models import agent as agent_model
|
||||||
|
from neutron.db import models_v2
|
||||||
|
|
||||||
|
|
||||||
def get_l3_agents():
|
def get_l3_agents():
|
||||||
@ -33,6 +34,13 @@ def get_l3_agents():
|
|||||||
return query.all()
|
return query.all()
|
||||||
|
|
||||||
|
|
||||||
|
def get_networks():
|
||||||
|
ctx = context.get_admin_context()
|
||||||
|
query = model_query.get_collection_query(ctx,
|
||||||
|
models_v2.Network)
|
||||||
|
return query.all()
|
||||||
|
|
||||||
|
|
||||||
class CoreChecks(base.BaseChecks):
|
class CoreChecks(base.BaseChecks):
|
||||||
|
|
||||||
def get_checks(self):
|
def get_checks(self):
|
||||||
@ -123,3 +131,28 @@ class CoreChecks(base.BaseChecks):
|
|||||||
return upgradecheck.Result(
|
return upgradecheck.Result(
|
||||||
upgradecheck.Code.SUCCESS,
|
upgradecheck.Code.SUCCESS,
|
||||||
_("L3 agents can use multiple networks as external gateways."))
|
_("L3 agents can use multiple networks as external gateways."))
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def network_mtu_check(checker):
|
||||||
|
if not cfg.CONF.database.connection:
|
||||||
|
return upgradecheck.Result(
|
||||||
|
upgradecheck.Code.WARNING,
|
||||||
|
_("Database connection string is not set. Check of 'mtu' in "
|
||||||
|
"networks can't be done"))
|
||||||
|
|
||||||
|
networks_with_empty_mtu_attr = []
|
||||||
|
for network in get_networks():
|
||||||
|
mtu = network.get('mtu', None)
|
||||||
|
if not mtu:
|
||||||
|
networks_with_empty_mtu_attr.append(network.get("id"))
|
||||||
|
|
||||||
|
if networks_with_empty_mtu_attr:
|
||||||
|
networks_list = ", ".join(networks_with_empty_mtu_attr)
|
||||||
|
return upgradecheck.Result(
|
||||||
|
upgradecheck.Code.WARNING,
|
||||||
|
_("The 'mtu' attribute of networks %s are not set "
|
||||||
|
"This attribute can't be null now.") % networks_list)
|
||||||
|
else:
|
||||||
|
return upgradecheck.Result(
|
||||||
|
upgradecheck.Code.SUCCESS,
|
||||||
|
_("The 'mtu' attribute of all networks are set."))
|
||||||
|
@ -404,7 +404,7 @@ class NeutronDbPluginV2(db_base_plugin_common.DbBasePluginCommon,
|
|||||||
args = {'tenant_id': n['tenant_id'],
|
args = {'tenant_id': n['tenant_id'],
|
||||||
'id': n.get('id') or uuidutils.generate_uuid(),
|
'id': n.get('id') or uuidutils.generate_uuid(),
|
||||||
'name': n['name'],
|
'name': n['name'],
|
||||||
'mtu': n.get('mtu'),
|
'mtu': n.get('mtu', constants.DEFAULT_NETWORK_MTU),
|
||||||
'admin_state_up': n['admin_state_up'],
|
'admin_state_up': n['admin_state_up'],
|
||||||
'status': n.get('status', constants.NET_STATUS_ACTIVE),
|
'status': n.get('status', constants.NET_STATUS_ACTIVE),
|
||||||
'description': n.get('description')}
|
'description': n.get('description')}
|
||||||
|
@ -1 +1 @@
|
|||||||
c3e9d13c4367
|
86274d77933e
|
||||||
|
@ -0,0 +1,55 @@
|
|||||||
|
# Copyright 2019 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
|
||||||
|
from neutron_lib import constants
|
||||||
|
import sqlalchemy as sa
|
||||||
|
|
||||||
|
|
||||||
|
"""change_mtu_to_not_null
|
||||||
|
|
||||||
|
Revision ID: 86274d77933e
|
||||||
|
Revises: c3e9d13c4367
|
||||||
|
Create Date: 2019-08-30 15:52:30.015146
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision = '86274d77933e'
|
||||||
|
down_revision = 'c3e9d13c4367'
|
||||||
|
|
||||||
|
|
||||||
|
networks = sa.Table(
|
||||||
|
'networks', sa.MetaData(),
|
||||||
|
sa.Column('id', sa.String(length=36), nullable=False),
|
||||||
|
sa.Column('mtu', sa.Integer(), nullable=True))
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade_existing_records():
|
||||||
|
session = sa.orm.Session(bind=op.get_bind())
|
||||||
|
with session.begin(subtransactions=True):
|
||||||
|
for row in session.query(networks):
|
||||||
|
if row[1] is None:
|
||||||
|
session.execute(networks.update().values(
|
||||||
|
mtu=constants.DEFAULT_NETWORK_MTU).where(
|
||||||
|
networks.c.id == row[0]))
|
||||||
|
session.commit()
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade():
|
||||||
|
upgrade_existing_records()
|
||||||
|
op.alter_column('networks', 'mtu', nullable=False,
|
||||||
|
server_default=str(constants.DEFAULT_NETWORK_MTU),
|
||||||
|
existing_type=sa.INTEGER())
|
@ -268,9 +268,9 @@ class Network(standard_attr.HasStandardAttributes, model_base.BASEV2,
|
|||||||
lazy='subquery',
|
lazy='subquery',
|
||||||
cascade='all, delete, delete-orphan')
|
cascade='all, delete, delete-orphan')
|
||||||
availability_zone_hints = sa.Column(sa.String(255))
|
availability_zone_hints = sa.Column(sa.String(255))
|
||||||
# TODO(ihrachys) provide data migration path to fill in mtus for existing
|
mtu = sa.Column(sa.Integer, nullable=False,
|
||||||
# networks in Queens when all controllers run Pike+ code
|
default=constants.DEFAULT_NETWORK_MTU,
|
||||||
mtu = sa.Column(sa.Integer, nullable=True)
|
server_default=str(constants.DEFAULT_NETWORK_MTU))
|
||||||
dhcp_agents = orm.relationship(
|
dhcp_agents = orm.relationship(
|
||||||
'Agent', lazy='subquery', viewonly=True,
|
'Agent', lazy='subquery', viewonly=True,
|
||||||
secondary=ndab_model.NetworkDhcpAgentBinding.__table__)
|
secondary=ndab_model.NetworkDhcpAgentBinding.__table__)
|
||||||
|
@ -14,7 +14,9 @@
|
|||||||
|
|
||||||
from neutron_lib.api.definitions import availability_zone as az_def
|
from neutron_lib.api.definitions import availability_zone as az_def
|
||||||
from neutron_lib.api.validators import availability_zone as az_validator
|
from neutron_lib.api.validators import availability_zone as az_validator
|
||||||
|
from neutron_lib import constants
|
||||||
from oslo_utils import versionutils
|
from oslo_utils import versionutils
|
||||||
|
from oslo_versionedobjects import exception
|
||||||
from oslo_versionedobjects import fields as obj_fields
|
from oslo_versionedobjects import fields as obj_fields
|
||||||
import sqlalchemy as sa
|
import sqlalchemy as sa
|
||||||
|
|
||||||
@ -214,7 +216,8 @@ class ExternalNetwork(base.NeutronDbObject):
|
|||||||
@base.NeutronObjectRegistry.register
|
@base.NeutronObjectRegistry.register
|
||||||
class Network(rbac_db.NeutronRbacObject):
|
class Network(rbac_db.NeutronRbacObject):
|
||||||
# Version 1.0: Initial version
|
# Version 1.0: Initial version
|
||||||
VERSION = '1.0'
|
# Version 1.1: Changed 'mtu' to be not nullable
|
||||||
|
VERSION = '1.1'
|
||||||
|
|
||||||
rbac_db_cls = NetworkRBAC
|
rbac_db_cls = NetworkRBAC
|
||||||
db_model = models_v2.Network
|
db_model = models_v2.Network
|
||||||
@ -231,7 +234,7 @@ class Network(rbac_db.NeutronRbacObject):
|
|||||||
nullable=True),
|
nullable=True),
|
||||||
'shared': obj_fields.BooleanField(default=False),
|
'shared': obj_fields.BooleanField(default=False),
|
||||||
|
|
||||||
'mtu': obj_fields.IntegerField(nullable=True),
|
'mtu': obj_fields.IntegerField(default=constants.DEFAULT_NETWORK_MTU),
|
||||||
|
|
||||||
# TODO(ihrachys): consider exposing availability zones
|
# TODO(ihrachys): consider exposing availability zones
|
||||||
|
|
||||||
@ -344,6 +347,14 @@ class Network(rbac_db.NeutronRbacObject):
|
|||||||
# TODO(ihrachys): provide actual implementation
|
# TODO(ihrachys): provide actual implementation
|
||||||
return set()
|
return set()
|
||||||
|
|
||||||
|
def obj_make_compatible(self, primitive, target_version):
|
||||||
|
_target_version = versionutils.convert_version_to_tuple(target_version)
|
||||||
|
if _target_version >= (1, 1):
|
||||||
|
if primitive['mtu'] is None:
|
||||||
|
# mtu will not be nullable after
|
||||||
|
raise exception.IncompatibleObjectVersion(
|
||||||
|
objver=target_version, objname=self.__class__.__name__)
|
||||||
|
|
||||||
|
|
||||||
@base.NeutronObjectRegistry.register
|
@base.NeutronObjectRegistry.register
|
||||||
class SegmentHostMapping(base.NeutronDbObject):
|
class SegmentHostMapping(base.NeutronDbObject):
|
||||||
|
@ -1144,12 +1144,6 @@ class Ml2Plugin(db_base_plugin_v2.NeutronDbPluginV2,
|
|||||||
with db_api.CONTEXT_WRITER.using(context):
|
with db_api.CONTEXT_WRITER.using(context):
|
||||||
net_db = self._get_network(context, id)
|
net_db = self._get_network(context, id)
|
||||||
|
|
||||||
# NOTE(ihrachys) pre Pike networks may have null mtus; update them
|
|
||||||
# in database if needed
|
|
||||||
# TODO(ihrachys) remove in Queens+ when mtu is not nullable
|
|
||||||
if net_db.mtu is None:
|
|
||||||
net_db.mtu = self._get_network_mtu(net_db, validate=False)
|
|
||||||
|
|
||||||
net_data = self._make_network_dict(net_db, context=context)
|
net_data = self._make_network_dict(net_db, context=context)
|
||||||
self.type_manager.extend_network_dict_provider(context, net_data)
|
self.type_manager.extend_network_dict_provider(context, net_data)
|
||||||
|
|
||||||
@ -1164,13 +1158,8 @@ class Ml2Plugin(db_base_plugin_v2.NeutronDbPluginV2,
|
|||||||
nets_db = super(Ml2Plugin, self)._get_networks(
|
nets_db = super(Ml2Plugin, self)._get_networks(
|
||||||
context, filters, None, sorts, limit, marker, page_reverse)
|
context, filters, None, sorts, limit, marker, page_reverse)
|
||||||
|
|
||||||
# NOTE(ihrachys) pre Pike networks may have null mtus; update them
|
|
||||||
# in database if needed
|
|
||||||
# TODO(ihrachys) remove in Queens+ when mtu is not nullable
|
|
||||||
net_data = []
|
net_data = []
|
||||||
for net in nets_db:
|
for net in nets_db:
|
||||||
if net.mtu is None:
|
|
||||||
net.mtu = self._get_network_mtu(net, validate=False)
|
|
||||||
net_data.append(self._make_network_dict(net, context=context))
|
net_data.append(self._make_network_dict(net, context=context))
|
||||||
|
|
||||||
self.type_manager.extend_networks_dict_provider(context, net_data)
|
self.type_manager.extend_networks_dict_provider(context, net_data)
|
||||||
@ -1241,12 +1230,6 @@ class Ml2Plugin(db_base_plugin_v2.NeutronDbPluginV2,
|
|||||||
result, net_db, ipam_sub = self._create_subnet_precommit(
|
result, net_db, ipam_sub = self._create_subnet_precommit(
|
||||||
context, subnet)
|
context, subnet)
|
||||||
|
|
||||||
# NOTE(ihrachys) pre Pike networks may have null mtus; update them
|
|
||||||
# in database if needed
|
|
||||||
# TODO(ihrachys) remove in Queens+ when mtu is not nullable
|
|
||||||
if net_db['mtu'] is None:
|
|
||||||
net_db['mtu'] = self._get_network_mtu(net_db, validate=False)
|
|
||||||
|
|
||||||
self.extension_manager.process_create_subnet(
|
self.extension_manager.process_create_subnet(
|
||||||
context, subnet[subnet_def.RESOURCE_NAME], result)
|
context, subnet[subnet_def.RESOURCE_NAME], result)
|
||||||
network = self._make_network_dict(net_db, context=context)
|
network = self._make_network_dict(net_db, context=context)
|
||||||
|
@ -106,3 +106,25 @@ class TestChecks(base.BaseTestCase):
|
|||||||
self.assertIn('Host B', result.details)
|
self.assertIn('Host B', result.details)
|
||||||
self.assertNotIn('Host A', result.details)
|
self.assertNotIn('Host A', result.details)
|
||||||
self.assertNotIn('Host C', result.details)
|
self.assertNotIn('Host C', result.details)
|
||||||
|
|
||||||
|
def test_network_mtu_check_good(self):
|
||||||
|
networks = [
|
||||||
|
{'id': 'net-uuid-a', 'mtu': 1500},
|
||||||
|
{'id': 'net-uuid-b', 'mtu': 1450}
|
||||||
|
]
|
||||||
|
with mock.patch.object(checks, "get_networks", return_value=networks):
|
||||||
|
result = checks.CoreChecks.network_mtu_check(
|
||||||
|
mock.Mock())
|
||||||
|
self.assertEqual(Code.SUCCESS, result.code)
|
||||||
|
|
||||||
|
def test_network_mtu_check_bad(self):
|
||||||
|
networks = [
|
||||||
|
{'id': 'net-uuid-a', 'mtu': None},
|
||||||
|
{'id': 'net-uuid-b', 'mtu': 1500},
|
||||||
|
]
|
||||||
|
with mock.patch.object(checks, "get_networks", return_value=networks):
|
||||||
|
result = checks.CoreChecks.network_mtu_check(
|
||||||
|
mock.Mock())
|
||||||
|
self.assertEqual(Code.WARNING, result.code)
|
||||||
|
self.assertIn('net-uuid-a', result.details)
|
||||||
|
self.assertNotIn('net-uuid-b', result.details)
|
||||||
|
@ -6503,7 +6503,8 @@ class DbModelMixin(object):
|
|||||||
with db_api.CONTEXT_WRITER.using(ctx):
|
with db_api.CONTEXT_WRITER.using(ctx):
|
||||||
network = models_v2.Network(name="net_net", status="OK",
|
network = models_v2.Network(name="net_net", status="OK",
|
||||||
admin_state_up=True,
|
admin_state_up=True,
|
||||||
project_id='fake_project')
|
project_id='fake_project',
|
||||||
|
mtu=1500)
|
||||||
ctx.session.add(network)
|
ctx.session.add(network)
|
||||||
with db_api.autonested_transaction(ctx.session):
|
with db_api.autonested_transaction(ctx.session):
|
||||||
sg = sg_models.SecurityGroup(name='sg', description='sg')
|
sg = sg_models.SecurityGroup(name='sg', description='sg')
|
||||||
@ -6516,7 +6517,8 @@ class DbModelMixin(object):
|
|||||||
project_id=network.project_id,
|
project_id=network.project_id,
|
||||||
target_tenant='*').create()
|
target_tenant='*').create()
|
||||||
net2 = models_v2.Network(name="net_net2", status="OK",
|
net2 = models_v2.Network(name="net_net2", status="OK",
|
||||||
admin_state_up=True)
|
admin_state_up=True,
|
||||||
|
mtu=1500)
|
||||||
ctx.session.add(net2)
|
ctx.session.add(net2)
|
||||||
pl = db_base_plugin_common.DbBasePluginCommon()
|
pl = db_base_plugin_common.DbBasePluginCommon()
|
||||||
self.assertTrue(pl._make_network_dict(network, context=ctx)['shared'])
|
self.assertTrue(pl._make_network_dict(network, context=ctx)['shared'])
|
||||||
|
@ -57,7 +57,7 @@ object_data = {
|
|||||||
'MeteringLabel': '1.0-cc4b620a3425222447cbe459f62de533',
|
'MeteringLabel': '1.0-cc4b620a3425222447cbe459f62de533',
|
||||||
'MeteringLabelRule': '1.0-b5c5717e7bab8d1af1623156012a5842',
|
'MeteringLabelRule': '1.0-b5c5717e7bab8d1af1623156012a5842',
|
||||||
'Log': '1.0-6391351c0f34ed34375a19202f361d24',
|
'Log': '1.0-6391351c0f34ed34375a19202f361d24',
|
||||||
'Network': '1.0-f2f6308f79731a767b92b26b0f4f3849',
|
'Network': '1.1-c3e9ecc0618ee934181d91b143a48901',
|
||||||
'NetworkDhcpAgentBinding': '1.1-d9443c88809ffa4c45a0a5a48134b54a',
|
'NetworkDhcpAgentBinding': '1.1-d9443c88809ffa4c45a0a5a48134b54a',
|
||||||
'NetworkDNSDomain': '1.0-420db7910294608534c1e2e30d6d8319',
|
'NetworkDNSDomain': '1.0-420db7910294608534c1e2e30d6d8319',
|
||||||
'NetworkPortSecurity': '1.0-b30802391a87945ee9c07582b4ff95e3',
|
'NetworkPortSecurity': '1.0-b30802391a87945ee9c07582b4ff95e3',
|
||||||
|
@ -0,0 +1,5 @@
|
|||||||
|
---
|
||||||
|
upgrade:
|
||||||
|
- |
|
||||||
|
The network ``mtu`` attribute is set to be non-nullable. If the ``mtu`` is
|
||||||
|
empty(create before Pike version), it is set to the default value of 1500.
|
Loading…
Reference in New Issue
Block a user