c22c89b104
The current code will retry over and over to reload the driver from stevedore if it fails to connect. That's not what we meant, we really just want to retry the connection part. Change-Id: I681029196f834999895a4d4186417c1da85a32ba
156 lines
5.6 KiB
Python
156 lines
5.6 KiB
Python
#
|
|
# Copyright 2012 New Dream Network, LLC (DreamHost)
|
|
#
|
|
# 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.
|
|
"""Storage backend management
|
|
"""
|
|
|
|
from oslo_config import cfg
|
|
from oslo_log import log
|
|
import retrying
|
|
import six.moves.urllib.parse as urlparse
|
|
from stevedore import driver
|
|
|
|
from aodh import utils
|
|
|
|
_NAMESPACE = 'aodh.storage'
|
|
|
|
|
|
LOG = log.getLogger(__name__)
|
|
|
|
OLD_OPTS = [
|
|
cfg.StrOpt('database_connection',
|
|
secret=True,
|
|
help='DEPRECATED - Database connection string.',
|
|
),
|
|
]
|
|
|
|
OPTS = [
|
|
cfg.IntOpt('alarm_history_time_to_live',
|
|
default=-1,
|
|
help=("Number of seconds that alarm histories are kept "
|
|
"in the database for (<= 0 means forever).")),
|
|
cfg.StrOpt('mongodb_replica_set',
|
|
default=None,
|
|
help='The name of the replica set which is used to connect to '
|
|
'MongoDB database. If it is set, MongoReplicaSetClient '
|
|
'will be used instead of MongoClient.'),
|
|
cfg.IntOpt('db2nosql_resource_id_maxlen',
|
|
default=512,
|
|
help="The max length of resources id in DB2 nosql, "
|
|
"the value should be larger than len(hostname) * 2 "
|
|
"as compute node's resource id is <hostname>_<nodename>."),
|
|
]
|
|
|
|
CLI_OPTS = [
|
|
cfg.BoolOpt('sql-expire-samples-only',
|
|
default=False,
|
|
help="Indicates if expirer expires only samples. If set true,"
|
|
" expired samples will be deleted, but residual"
|
|
" resource and meter definition data will remain.",
|
|
),
|
|
]
|
|
|
|
|
|
class StorageUnknownWriteError(Exception):
|
|
"""Error raised when an unknown error occurs while recording."""
|
|
|
|
|
|
class StorageBadVersion(Exception):
|
|
"""Error raised when the storage backend version is not good enough."""
|
|
|
|
|
|
class StorageBadAggregate(Exception):
|
|
"""Error raised when an aggregate is unacceptable to storage backend."""
|
|
code = 400
|
|
|
|
|
|
def get_connection_from_config(conf):
|
|
retries = conf.database.max_retries
|
|
url = conf.database.connection
|
|
connection_scheme = urlparse.urlparse(url).scheme
|
|
# SQLAlchemy connections specify may specify a 'dialect' or
|
|
# 'dialect+driver'. Handle the case where driver is specified.
|
|
engine_name = connection_scheme.split('+')[0]
|
|
# NOTE: translation not applied bug #1446983
|
|
LOG.debug('looking for %(name)r driver in %(namespace)r',
|
|
{'name': engine_name, 'namespace': _NAMESPACE})
|
|
mgr = driver.DriverManager(_NAMESPACE, engine_name)
|
|
|
|
# Convert retry_interval secs to msecs for retry decorator
|
|
@retrying.retry(wait_fixed=conf.database.retry_interval * 1000,
|
|
stop_max_attempt_number=retries if retries >= 0 else None)
|
|
def _get_connection():
|
|
"""Return an open connection to the database."""
|
|
return mgr.driver(conf, url)
|
|
|
|
return _get_connection()
|
|
|
|
|
|
class SampleFilter(object):
|
|
"""Holds the properties for building a query from a meter/sample filter.
|
|
|
|
:param user: The sample owner.
|
|
:param project: The sample project.
|
|
:param start_timestamp: Earliest time point in the request.
|
|
:param start_timestamp_op: Earliest timestamp operation in the request.
|
|
:param end_timestamp: Latest time point in the request.
|
|
:param end_timestamp_op: Latest timestamp operation in the request.
|
|
:param resource: Optional filter for resource id.
|
|
:param meter: Optional filter for meter type using the meter name.
|
|
:param source: Optional source filter.
|
|
:param message_id: Optional sample_id filter.
|
|
:param metaquery: Optional filter on the metadata
|
|
"""
|
|
def __init__(self, user=None, project=None,
|
|
start_timestamp=None, start_timestamp_op=None,
|
|
end_timestamp=None, end_timestamp_op=None,
|
|
resource=None, meter=None,
|
|
source=None, message_id=None,
|
|
metaquery=None):
|
|
self.user = user
|
|
self.project = project
|
|
self.start_timestamp = utils.sanitize_timestamp(start_timestamp)
|
|
self.start_timestamp_op = start_timestamp_op
|
|
self.end_timestamp = utils.sanitize_timestamp(end_timestamp)
|
|
self.end_timestamp_op = end_timestamp_op
|
|
self.resource = resource
|
|
self.meter = meter
|
|
self.source = source
|
|
self.metaquery = metaquery or {}
|
|
self.message_id = message_id
|
|
|
|
def __repr__(self):
|
|
return ("<SampleFilter(user: %s,"
|
|
" project: %s,"
|
|
" start_timestamp: %s,"
|
|
" start_timestamp_op: %s,"
|
|
" end_timestamp: %s,"
|
|
" end_timestamp_op: %s,"
|
|
" resource: %s,"
|
|
" meter: %s,"
|
|
" source: %s,"
|
|
" metaquery: %s,"
|
|
" message_id: %s)>" %
|
|
(self.user,
|
|
self.project,
|
|
self.start_timestamp,
|
|
self.start_timestamp_op,
|
|
self.end_timestamp,
|
|
self.end_timestamp_op,
|
|
self.resource,
|
|
self.meter,
|
|
self.source,
|
|
self.metaquery,
|
|
self.message_id))
|