diff --git a/ceilometer/storage/impl_mongodb.py b/ceilometer/storage/impl_mongodb.py index 4b6949294..16f9a25c2 100644 --- a/ceilometer/storage/impl_mongodb.py +++ b/ceilometer/storage/impl_mongodb.py @@ -24,9 +24,7 @@ import copy import datetime import operator import os -import re import time -import urlparse import uuid import bson.code @@ -36,7 +34,6 @@ import pymongo from oslo.config import cfg from ceilometer.openstack.common import log -from ceilometer.openstack.common import network_utils from ceilometer.openstack.common import timeutils from ceilometer import storage from ceilometer.storage import base @@ -147,20 +144,15 @@ class ConnectionPool(object): def __init__(self): self._pool = {} - def connect(self, opts): - # opts is a dict, dict are unhashable, convert to tuple - connection_pool_key = tuple(sorted(opts.items())) - - if connection_pool_key not in self._pool: - LOG.info('connecting to MongoDB replicaset "%s" on %s', - opts['replica_set'], - opts['netloc']) - self._pool[connection_pool_key] = pymongo.Connection( - opts['netloc'], - replicaSet=opts['replica_set'], + def connect(self, url): + if url not in self._pool: + LOG.info('connecting to MongoDB on %s', url) + self._pool[url] = pymongo.MongoClient( + url, + use_greenlets=True, safe=True) - return self._pool.get(connection_pool_key) + return self._pool.get(url) class Connection(base.Connection): @@ -249,25 +241,23 @@ class Connection(base.Connection): }""") def __init__(self, conf): - opts = self._parse_connection_url(conf.database.connection) + url = conf.database.connection - if opts['netloc'] == '__test__': + if url == 'mongodb://__test__': url = os.environ.get('CEILOMETER_TEST_MONGODB_URL') if not url: raise RuntimeError( "No MongoDB test URL set," "export CEILOMETER_TEST_MONGODB_URL environment variable") - opts = self._parse_connection_url(url) # NOTE(jd) Use our own connection pooling on top of the Pymongo one. # We need that otherwise we overflow the MongoDB instance with new # connection since we instanciate a Pymongo client each time someone # requires a new storage connection. - self.conn = self.CONNECTION_POOL.connect(opts) + self.conn = self.CONNECTION_POOL.connect(url) - self.db = getattr(self.conn, opts['dbname']) - if 'username' in opts: - self.db.authenticate(opts['username'], opts['password']) + connection_options = pymongo.uri_parser.parse_uri(url) + self.db = getattr(self.conn, connection_options['database']) # NOTE(jd) Upgrading is just about creating index, so let's do this # on connection to be sure at least the TTL is correcly updated if @@ -333,21 +323,6 @@ class Connection(base.Connection): def clear(self): self.conn.drop_database(self.db) - @staticmethod - def _parse_connection_url(url): - opts = {} - result = network_utils.urlsplit(url) - opts['replica_set'] = urlparse.parse_qs( - result.query).get('replica_set', [""])[0] - opts['dbtype'] = result.scheme - opts['dbname'] = result.path.replace('/', '') - netloc_match = re.match(r'(?:(\w+:\w+)@)?(.*)', result.netloc) - auth = netloc_match.group(1) - opts['netloc'] = netloc_match.group(2) - if auth: - opts['username'], opts['password'] = auth.split(':') - return opts - def record_metering_data(self, data): """Write the data to the backend storage system. diff --git a/requirements.txt b/requirements.txt index 5055189fe..00fd91b7b 100644 --- a/requirements.txt +++ b/requirements.txt @@ -8,7 +8,7 @@ SQLAlchemy>=0.7,<=0.7.99 sqlalchemy-migrate>=0.7.2 alembic>=0.4.1 netaddr -pymongo>=2.2 +pymongo>=2.4 eventlet anyjson>=0.3.3 Flask==0.9 diff --git a/tests/storage/test_impl_mongodb.py b/tests/storage/test_impl_mongodb.py index eb2856cda..5449272e5 100644 --- a/tests/storage/test_impl_mongodb.py +++ b/tests/storage/test_impl_mongodb.py @@ -48,9 +48,10 @@ class MongoDBConnection(MongoDBEngineTestBase): impl_mongodb.Connection(cfg.CONF).conn) def test_replica_set(self): + # FIXME(Alexei_987) should not hardcode URL here cfg.CONF.set_override( 'connection', - 'mongodb://__test__?replica_set=foobar', + 'mongodb://localhost:29000/ceilometer?replicaSet=foobar', group='database') conn = impl_mongodb.Connection(cfg.CONF) self.assertTrue(conn.conn)