vmware-nsx/vmware_nsx/db/vnic_index_db.py
Gary Kotton e51421f5ca NSX|V: don't throw exception when same vnic is configured
In the event of a nova retry we should not throw an exception for
for a port update

Change-Id: Ic4df033952e3dddd170e0fb7b57499750a6155b0
Closes-bug: #1606809
2016-07-27 06:58:50 -07:00

104 lines
4.2 KiB
Python

# Copyright 2014 VMware, 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 sqlalchemy.orm import exc
from neutron.api.v2 import attributes as attr
from neutron.db import db_base_plugin_v2
from oslo_db import exception as db_exc
from oslo_log import log as logging
from vmware_nsx.db import nsxv_models
from vmware_nsx.extensions import vnicindex as vnicidx
LOG = logging.getLogger(__name__)
class VnicIndexDbMixin(object):
def _extend_port_vnic_index_binding(self, port_res, port_db):
state = port_db.vnic_index
port_res[vnicidx.VNIC_INDEX] = state.index if state else None
db_base_plugin_v2.NeutronDbPluginV2.register_dict_extend_funcs(
attr.PORTS, ['_extend_port_vnic_index_binding'])
def _get_port_vnic_index(self, context, port_id):
"""Returns the vnic index for the given port.
If the port is not associated with any vnic then return None
"""
session = context.session
try:
mapping = (session.query(nsxv_models.NsxvPortIndexMapping).
filter_by(port_id=port_id).one())
return mapping['index']
except exc.NoResultFound:
LOG.debug("No record in DB for vnic-index of port %s", port_id)
def _get_mappings_for_device_id(self, context, device_id):
session = context.session
mappings = (session.query(nsxv_models.NsxvPortIndexMapping).
filter_by(device_id=device_id))
return mappings
def _create_port_vnic_index_mapping(self, context, port_id,
device_id, index):
"""Save the port vnic-index to DB."""
session = context.session
with session.begin(subtransactions=True):
index_mapping_model = nsxv_models.NsxvPortIndexMapping(
port_id=port_id, device_id=device_id, index=index)
session.add(index_mapping_model)
def _update_port_vnic_index_mapping(self, context, port_id,
device_id, index):
session = context.session
# delete original entry
query = (session.query(nsxv_models.NsxvPortIndexMapping).
filter_by(device_id=device_id, index=index))
query.delete()
# create a new one
self._create_port_vnic_index_mapping(context, port_id, device_id,
index)
def _set_port_vnic_index_mapping(self, context, port_id, device_id, index):
"""Save the port vnic-index to DB."""
try:
self._create_port_vnic_index_mapping(context, port_id,
device_id, index)
except db_exc.DBDuplicateEntry:
# A retry for the nova scheduling could result in this error.
LOG.debug("Entry already exists for %s %s %s", port_id,
device_id, index)
mappings = self._get_mappings_for_device_id(context, device_id)
for mapping in mappings:
if (mapping['port_id'] != port_id and
mapping['index'] == index):
# a new port is using this device - update!
self._update_port_vnic_index_mapping(context, port_id,
device_id, index)
return
if (mapping['port_id'] == port_id and
mapping['index'] != index):
raise
def _delete_port_vnic_index_mapping(self, context, port_id):
"""Delete the port vnic-index association."""
session = context.session
query = (session.query(nsxv_models.NsxvPortIndexMapping).
filter_by(port_id=port_id))
query.delete()