de5ed2115a
To follow latest community practices we need to use oslo_log in murano instead of handcrafted libs. Unsupported log level 'audit' was changed to 'info'. Change-Id: I40c0f3790e34bf80d0b63554e86b3cbc019eefca
94 lines
2.6 KiB
Python
94 lines
2.6 KiB
Python
# Copyright (c) 2014 Mirantis, Inc.
|
|
#
|
|
# 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.
|
|
|
|
"""Session management functions."""
|
|
import threading
|
|
|
|
from oslo_config import cfg
|
|
from oslo_db import exception
|
|
from oslo_db import options
|
|
from oslo_db.sqlalchemy import session as db_session
|
|
from oslo_log import log as logging
|
|
from oslo_utils import timeutils
|
|
|
|
from murano.db.models import Lock
|
|
|
|
LOG = logging.getLogger(__name__)
|
|
CONF = cfg.CONF
|
|
|
|
options.set_defaults(CONF)
|
|
|
|
_FACADE = None
|
|
_LOCK = threading.Lock()
|
|
|
|
MAX_LOCK_RETRIES = 10
|
|
|
|
|
|
def _create_facade_lazily():
|
|
global _LOCK, _FACADE
|
|
|
|
if _FACADE is None:
|
|
with _LOCK:
|
|
if _FACADE is None:
|
|
_FACADE = db_session.EngineFacade.from_config(CONF,
|
|
sqlite_fk=True)
|
|
return _FACADE
|
|
|
|
|
|
def get_engine():
|
|
facade = _create_facade_lazily()
|
|
return facade.get_engine()
|
|
|
|
|
|
def get_session(**kwargs):
|
|
facade = _create_facade_lazily()
|
|
return facade.get_session(**kwargs)
|
|
|
|
|
|
def get_lock(name, session=None):
|
|
if session is None:
|
|
session = get_session()
|
|
nested = False
|
|
else:
|
|
nested = session.transaction is not None
|
|
return _get_or_create_lock(name, session, nested)
|
|
|
|
|
|
def _get_or_create_lock(name, session, nested, retry=0):
|
|
if nested:
|
|
session.begin_nested()
|
|
else:
|
|
session.begin()
|
|
existing = session.query(Lock).get(name)
|
|
if existing is None:
|
|
try:
|
|
# no lock found, creating a new one
|
|
lock = Lock(id=name, ts=timeutils.utcnow())
|
|
lock.save(session)
|
|
return session.transaction
|
|
# lock created and acquired
|
|
except exception.DBDuplicateEntry:
|
|
session.rollback()
|
|
if retry >= MAX_LOCK_RETRIES:
|
|
raise
|
|
else:
|
|
# other transaction has created a lock, repeat to acquire
|
|
# via update
|
|
return _get_or_create_lock(name, session, nested, retry + 1)
|
|
else:
|
|
# lock found, acquiring by doing update
|
|
existing.ts = timeutils.utcnow()
|
|
existing.save(session)
|
|
return session.transaction
|