[OVN] Add revision number methods
OVNRevisionNumbers registers are used to check the synchronization status of the objects handled by OVN, assigning a revision number to those objects stored in the Neutron DB, also present in the OVN one. The methods implemented are: - create_initial_revision - delete_revision - get_revision_row - bump_revision Partially-Implements: blueprint neutron-ovn-merge Change-Id: I8a3bf9a915f6812db6e78a189127613a2fa204ae
This commit is contained in:
parent
97e495671f
commit
04f693bbbb
166
neutron/db/ovn_revision_numbers_db.py
Normal file
166
neutron/db/ovn_revision_numbers_db.py
Normal file
@ -0,0 +1,166 @@
|
|||||||
|
# Copyright 2017 Red Hat, Inc.
|
||||||
|
# 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 neutron_lib.db import api as db_api
|
||||||
|
from neutron_lib import exceptions as n_exc
|
||||||
|
from oslo_config import cfg
|
||||||
|
from oslo_log import log
|
||||||
|
from sqlalchemy.orm import exc
|
||||||
|
|
||||||
|
from neutron.db.models import l3 # noqa
|
||||||
|
from neutron.db.models import ovn as ovn_models
|
||||||
|
from neutron.db.models import securitygroup # noqa
|
||||||
|
from neutron.db import models_v2 # noqa
|
||||||
|
from neutron.db import standard_attr
|
||||||
|
|
||||||
|
LOG = log.getLogger(__name__)
|
||||||
|
CONF = cfg.CONF
|
||||||
|
|
||||||
|
STD_ATTR_MAP = standard_attr.get_standard_attr_resource_model_map()
|
||||||
|
|
||||||
|
# NOTE(ralonsoh): to be moved to neutron-lib
|
||||||
|
TYPE_NETWORKS = 'networks'
|
||||||
|
TYPE_PORTS = 'ports'
|
||||||
|
TYPE_SECURITY_GROUP_RULES = 'security_group_rules'
|
||||||
|
TYPE_ROUTERS = 'routers'
|
||||||
|
TYPE_ROUTER_PORTS = 'router_ports'
|
||||||
|
TYPE_SECURITY_GROUPS = 'security_groups'
|
||||||
|
TYPE_FLOATINGIPS = 'floatingips'
|
||||||
|
TYPE_SUBNETS = 'subnets'
|
||||||
|
TYPES_OVN = (TYPE_NETWORKS, TYPE_PORTS, TYPE_SECURITY_GROUP_RULES,
|
||||||
|
TYPE_ROUTERS, TYPE_ROUTER_PORTS, TYPE_SECURITY_GROUPS,
|
||||||
|
TYPE_FLOATINGIPS, TYPE_SUBNETS)
|
||||||
|
INITIAL_REV_NUM = -1
|
||||||
|
|
||||||
|
|
||||||
|
# 1:2 mapping for OVN, neutron router ports are simple ports, but
|
||||||
|
# for OVN we handle LSP & LRP objects
|
||||||
|
if STD_ATTR_MAP:
|
||||||
|
STD_ATTR_MAP[TYPE_ROUTER_PORTS] = STD_ATTR_MAP[TYPE_PORTS]
|
||||||
|
|
||||||
|
|
||||||
|
# NOTE(ralonsoh): to be moved to neutron-lib
|
||||||
|
class StandardAttributeIDNotFound(n_exc.NeutronException):
|
||||||
|
message = 'Standard attribute ID not found for %(resource_uuid)s'
|
||||||
|
|
||||||
|
|
||||||
|
# NOTE(ralonsoh): to be moved to neutron-lib
|
||||||
|
class UnknownResourceType(n_exc.NeutronException):
|
||||||
|
message = 'Uknown resource type: %(resource_type)s'
|
||||||
|
|
||||||
|
|
||||||
|
def get_revision_number(resource, resource_type):
|
||||||
|
"""Get the resource's revision number based on its type."""
|
||||||
|
if resource_type in TYPES_OVN:
|
||||||
|
return resource['revision_number']
|
||||||
|
raise UnknownResourceType(resource_type=resource_type)
|
||||||
|
|
||||||
|
|
||||||
|
def _get_standard_attr_id(context, resource_uuid, resource_type):
|
||||||
|
try:
|
||||||
|
row = context.session.query(STD_ATTR_MAP[resource_type]).filter_by(
|
||||||
|
id=resource_uuid).one()
|
||||||
|
return row.standard_attr_id
|
||||||
|
except exc.NoResultFound:
|
||||||
|
raise StandardAttributeIDNotFound(resource_uuid=resource_uuid)
|
||||||
|
|
||||||
|
|
||||||
|
@db_api.retry_if_session_inactive()
|
||||||
|
def create_initial_revision(context, resource_uuid, resource_type,
|
||||||
|
revision_number=INITIAL_REV_NUM,
|
||||||
|
may_exist=False):
|
||||||
|
LOG.debug('create_initial_revision uuid=%s, type=%s, rev=%s',
|
||||||
|
resource_uuid, resource_type, revision_number)
|
||||||
|
db_func = context.session.merge if may_exist else context.session.add
|
||||||
|
with db_api.CONTEXT_WRITER.using(context):
|
||||||
|
std_attr_id = _get_standard_attr_id(
|
||||||
|
context, resource_uuid, resource_type)
|
||||||
|
row = ovn_models.OVNRevisionNumbers(
|
||||||
|
resource_uuid=resource_uuid, resource_type=resource_type,
|
||||||
|
standard_attr_id=std_attr_id, revision_number=revision_number)
|
||||||
|
db_func(row)
|
||||||
|
context.session.flush()
|
||||||
|
|
||||||
|
|
||||||
|
@db_api.retry_if_session_inactive()
|
||||||
|
def delete_revision(context, resource_uuid, resource_type):
|
||||||
|
LOG.debug('delete_revision(%s)', resource_uuid)
|
||||||
|
with db_api.CONTEXT_WRITER.using(context):
|
||||||
|
row = context.session.query(ovn_models.OVNRevisionNumbers).filter_by(
|
||||||
|
resource_uuid=resource_uuid,
|
||||||
|
resource_type=resource_type).one_or_none()
|
||||||
|
if row:
|
||||||
|
context.session.delete(row)
|
||||||
|
|
||||||
|
|
||||||
|
def _ensure_revision_row_exist(context, resource, resource_type):
|
||||||
|
"""Ensure the revision row exists.
|
||||||
|
|
||||||
|
Ensure the revision row exist before we try to bump its revision
|
||||||
|
number. This method is part of the migration plan to deal with
|
||||||
|
resources that have been created prior to the database sync work
|
||||||
|
getting merged.
|
||||||
|
"""
|
||||||
|
# TODO(lucasagomes): As the docstring says, this method was created to
|
||||||
|
# deal with objects that already existed before the sync work. I believe
|
||||||
|
# that we can remove this method after few development cycles. Or,
|
||||||
|
# if we decide to make a migration script as well.
|
||||||
|
with db_api.CONTEXT_WRITER.using(context):
|
||||||
|
if not context.session.query(ovn_models.OVNRevisionNumbers).filter_by(
|
||||||
|
resource_uuid=resource['id'],
|
||||||
|
resource_type=resource_type).one_or_none():
|
||||||
|
LOG.warning(
|
||||||
|
'No revision row found for %(res_uuid)s (type: '
|
||||||
|
'%(res_type)s) when bumping the revision number. '
|
||||||
|
'Creating one.', {'res_uuid': resource['id'],
|
||||||
|
'res_type': resource_type})
|
||||||
|
create_initial_revision(context, resource['id'], resource_type)
|
||||||
|
|
||||||
|
|
||||||
|
@db_api.retry_if_session_inactive()
|
||||||
|
def get_revision_row(context, resource_uuid):
|
||||||
|
try:
|
||||||
|
with db_api.CONTEXT_READER.using(context):
|
||||||
|
return context.session.query(
|
||||||
|
ovn_models.OVNRevisionNumbers).filter_by(
|
||||||
|
resource_uuid=resource_uuid).one()
|
||||||
|
except exc.NoResultFound:
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
@db_api.retry_if_session_inactive()
|
||||||
|
def bump_revision(context, resource, resource_type):
|
||||||
|
revision_number = get_revision_number(resource, resource_type)
|
||||||
|
with db_api.CONTEXT_WRITER.using(context):
|
||||||
|
_ensure_revision_row_exist(context, resource, resource_type)
|
||||||
|
std_attr_id = _get_standard_attr_id(
|
||||||
|
context, resource['id'], resource_type)
|
||||||
|
row = context.session.merge(ovn_models.OVNRevisionNumbers(
|
||||||
|
standard_attr_id=std_attr_id, resource_uuid=resource['id'],
|
||||||
|
resource_type=resource_type))
|
||||||
|
if revision_number < row.revision_number:
|
||||||
|
LOG.debug(
|
||||||
|
'Skip bumping the revision number for %(res_uuid)s (type: '
|
||||||
|
'%(res_type)s) to %(rev_num)d. A higher version is already '
|
||||||
|
'registered in the database (%(new_rev)d)',
|
||||||
|
{'res_type': resource_type, 'res_uuid': resource['id'],
|
||||||
|
'rev_num': revision_number, 'new_rev': row.revision_number})
|
||||||
|
return
|
||||||
|
row.revision_number = revision_number
|
||||||
|
context.session.merge(row)
|
||||||
|
LOG.info('Successfully bumped revision number for resource '
|
||||||
|
'%(res_uuid)s (type: %(res_type)s) to %(rev_num)d',
|
||||||
|
{'res_uuid': resource['id'], 'res_type': resource_type,
|
||||||
|
'rev_num': revision_number})
|
95
neutron/tests/unit/db/test_ovn_revision_numbers_db.py
Normal file
95
neutron/tests/unit/db/test_ovn_revision_numbers_db.py
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
# Copyright 2019 Red Hat, Inc.
|
||||||
|
# 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.
|
||||||
|
|
||||||
|
import mock
|
||||||
|
|
||||||
|
from neutron_lib import context
|
||||||
|
from neutron_lib.db import api as db_api
|
||||||
|
from oslo_db import exception as db_exc
|
||||||
|
|
||||||
|
from neutron.db.models import ovn as ovn_models
|
||||||
|
from neutron.db import ovn_revision_numbers_db as ovn_rn_db
|
||||||
|
from neutron.tests.unit.db import test_db_base_plugin_v2
|
||||||
|
|
||||||
|
|
||||||
|
class TestRevisionNumber(test_db_base_plugin_v2.NeutronDbPluginV2TestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestRevisionNumber, self).setUp()
|
||||||
|
self.ctx = context.get_admin_context()
|
||||||
|
self.addCleanup(self._delete_objs)
|
||||||
|
res = self._create_network(fmt=self.fmt, name='net',
|
||||||
|
admin_state_up=True)
|
||||||
|
self.net = self.deserialize(self.fmt, res)['network']
|
||||||
|
|
||||||
|
def _delete_objs(self):
|
||||||
|
with db_api.CONTEXT_WRITER.using(self.ctx):
|
||||||
|
self.ctx.session.query(
|
||||||
|
ovn_models.OVNRevisionNumbers).delete()
|
||||||
|
|
||||||
|
def _create_initial_revision(self, resource_uuid, resource_type,
|
||||||
|
revision_number=ovn_rn_db.INITIAL_REV_NUM,
|
||||||
|
may_exist=False):
|
||||||
|
with self.ctx.session.begin(subtransactions=True):
|
||||||
|
ovn_rn_db.create_initial_revision(
|
||||||
|
self.ctx, resource_uuid, resource_type,
|
||||||
|
revision_number=revision_number, may_exist=may_exist)
|
||||||
|
|
||||||
|
def test_bump_revision(self):
|
||||||
|
self._create_initial_revision(self.net['id'], ovn_rn_db.TYPE_NETWORKS)
|
||||||
|
self.net['revision_number'] = 123
|
||||||
|
ovn_rn_db.bump_revision(self.ctx, self.net,
|
||||||
|
ovn_rn_db.TYPE_NETWORKS)
|
||||||
|
row = ovn_rn_db.get_revision_row(self.ctx, self.net['id'])
|
||||||
|
self.assertEqual(123, row.revision_number)
|
||||||
|
|
||||||
|
def test_bump_older_revision(self):
|
||||||
|
self._create_initial_revision(self.net['id'], ovn_rn_db.TYPE_NETWORKS,
|
||||||
|
revision_number=124)
|
||||||
|
self.net['revision_number'] = 1
|
||||||
|
ovn_rn_db.bump_revision(self.ctx, self.net,
|
||||||
|
ovn_rn_db.TYPE_NETWORKS)
|
||||||
|
row = ovn_rn_db.get_revision_row(self.ctx, self.net['id'])
|
||||||
|
self.assertEqual(124, row.revision_number)
|
||||||
|
|
||||||
|
@mock.patch.object(ovn_rn_db.LOG, 'warning')
|
||||||
|
def test_bump_revision_row_not_found(self, mock_log):
|
||||||
|
self.net['revision_number'] = 123
|
||||||
|
ovn_rn_db.bump_revision(self.ctx, self.net, ovn_rn_db.TYPE_NETWORKS)
|
||||||
|
# Assert the revision number wasn't bumped
|
||||||
|
row = ovn_rn_db.get_revision_row(self.ctx, self.net['id'])
|
||||||
|
self.assertEqual(123, row.revision_number)
|
||||||
|
self.assertIn('No revision row found for', mock_log.call_args[0][0])
|
||||||
|
|
||||||
|
def test_delete_revision(self):
|
||||||
|
self._create_initial_revision(self.net['id'], ovn_rn_db.TYPE_NETWORKS)
|
||||||
|
ovn_rn_db.delete_revision(self.ctx, self.net['id'],
|
||||||
|
ovn_rn_db.TYPE_NETWORKS)
|
||||||
|
row = ovn_rn_db.get_revision_row(self.ctx, self.net['id'])
|
||||||
|
self.assertIsNone(row)
|
||||||
|
|
||||||
|
def test_create_initial_revision_may_exist_duplicated_entry(self):
|
||||||
|
args = (self.net['id'], ovn_rn_db.TYPE_NETWORKS)
|
||||||
|
self._create_initial_revision(*args)
|
||||||
|
|
||||||
|
# Assert DBDuplicateEntry is raised when may_exist is False (default)
|
||||||
|
self.assertRaises(db_exc.DBDuplicateEntry,
|
||||||
|
self._create_initial_revision, *args)
|
||||||
|
|
||||||
|
try:
|
||||||
|
self._create_initial_revision(*args, may_exist=True)
|
||||||
|
except db_exc.DBDuplicateEntry:
|
||||||
|
self.fail("create_initial_revision shouldn't raise "
|
||||||
|
"DBDuplicateEntry when may_exist is True")
|
Loading…
x
Reference in New Issue
Block a user