added mongodb auth

Implements bug 1022679.  mongodb needs auth=true to be turned on, and user
accounts in the appropriate dbs would need to be created.  Add to ceilometer
conf:

'database_connection=mongodb://username:password@<hostname>:<port#>/<dbname>'

Change-Id: I6fbe563cb5660e2374edfe39b325a68c56ccd39a
This commit is contained in:
John H. Tran 2012-08-28 09:37:13 -07:00
parent 5b153d5a8c
commit 480052fc16
5 changed files with 42 additions and 38 deletions

View File

@ -23,14 +23,16 @@ import pkg_resources
from ceilometer.openstack.common import log from ceilometer.openstack.common import log
from ceilometer.openstack.common import cfg from ceilometer.openstack.common import cfg
from urlparse import urlparse
LOG = log.getLogger(__name__) LOG = log.getLogger(__name__)
STORAGE_ENGINE_NAMESPACE = 'ceilometer.storage' STORAGE_ENGINE_NAMESPACE = 'ceilometer.storage'
STORAGE_OPTS = [ STORAGE_OPTS = [
cfg.StrOpt('metering_storage_engine', cfg.StrOpt('database_connection',
default='mongodb', default='mongodb://localhost:27017/ceilometer',
help='The name of the storage engine to use', help='Database connection string',
), ),
] ]
@ -48,7 +50,7 @@ def register_opts(conf):
def get_engine(conf): def get_engine(conf):
"""Load the configured engine and return an instance. """Load the configured engine and return an instance.
""" """
engine_name = conf.metering_storage_engine engine_name = urlparse(conf.database_connection).scheme
LOG.debug('looking for %r driver in %r', LOG.debug('looking for %r driver in %r',
engine_name, STORAGE_ENGINE_NAMESPACE) engine_name, STORAGE_ENGINE_NAMESPACE)
for ep in pkg_resources.iter_entry_points(STORAGE_ENGINE_NAMESPACE, for ep in pkg_resources.iter_entry_points(STORAGE_ENGINE_NAMESPACE,

View File

@ -27,6 +27,9 @@ from ceilometer.storage import base
import bson.code import bson.code
import pymongo import pymongo
import re
from urlparse import urlparse
LOG = log.getLogger(__name__) LOG = log.getLogger(__name__)
@ -57,20 +60,7 @@ class MongoDBStorage(base.StorageEngine):
} }
""" """
OPTIONS = [ OPTIONS = []
cfg.StrOpt('mongodb_dbname',
default='ceilometer',
help='Database name',
),
cfg.StrOpt('mongodb_host',
default='localhost',
help='hostname or IP of server running MongoDB',
),
cfg.IntOpt('mongodb_port',
default=27017,
help='port number where MongoDB is running',
),
]
def register_opts(self, conf): def register_opts(self, conf):
"""Register any configuration options used by this engine. """Register any configuration options used by this engine.
@ -169,10 +159,12 @@ class Connection(base.Connection):
""") """)
def __init__(self, conf): def __init__(self, conf):
LOG.info('connecting to MongoDB on %s:%s', opts = self._parse_connection_url(conf.database_connection)
conf.mongodb_host, conf.mongodb_port) LOG.info('connecting to MongoDB on %s:%s', opts['host'], opts['port'])
self.conn = self._get_connection(conf) self.conn = self._get_connection(opts)
self.db = getattr(self.conn, conf.mongodb_dbname) self.db = getattr(self.conn, opts['dbname'])
if 'username' in opts:
self.db.authenticate(opts['username'], opts['password'])
# Establish indexes # Establish indexes
# #
@ -195,7 +187,7 @@ class Connection(base.Connection):
]) ])
return return
def _get_connection(self, conf): def _get_connection(self, opts):
"""Return a connection to the database. """Return a connection to the database.
.. note:: .. note::
@ -203,10 +195,25 @@ class Connection(base.Connection):
The tests use a subclass to override this and return an The tests use a subclass to override this and return an
in-memory connection. in-memory connection.
""" """
return pymongo.Connection(conf.mongodb_host, return pymongo.Connection(opts['host'], opts['port'], safe=True)
conf.mongodb_port,
safe=True, def _parse_connection_url(self, url):
) opts = {}
result = urlparse(url)
opts['dbtype'] = result.scheme
opts['dbname'] = result.path.replace('/', '')
netloc_match = re.match(r'(?:(\w+:\w+)@)?(.*)', result.netloc)
auth = netloc_match.group(1)
netloc = netloc_match.group(2)
if auth:
opts['username'], opts['password'] = auth.split(':')
if ':' in netloc:
opts['host'], port = netloc.split(':')
else:
opts['host'] = netloc
port = 27017
opts['port'] = port and int(port) or 27017
return opts
def record_metering_data(self, data): def record_metering_data(self, data):
"""Write the data to the backend storage system. """Write the data to the backend storage system.

View File

@ -61,9 +61,7 @@ class TestBase(unittest.TestCase):
self.test_app = self.app.test_client() self.test_app = self.app.test_client()
self.conf = mock.Mock() self.conf = mock.Mock()
self.conf.metering_storage_engine = 'mongodb' self.conf.metering_storage_engine = 'mongodb'
self.conf.mongodb_host = 'localhost' self.conf.database_connection = 'mongodb://localhost/%s' % self.DBNAME
self.conf.mongodb_port = 27017
self.conf.mongodb_dbname = self.DBNAME
self.conn = Connection(self.conf) self.conn = Connection(self.conf)
self.conn.conn.drop_database(self.DBNAME) self.conn.conn.drop_database(self.DBNAME)
self.conn.conn[self.DBNAME] self.conn.conn[self.DBNAME]

View File

@ -26,14 +26,14 @@ from ceilometer.storage import impl_log
def test_get_engine(): def test_get_engine():
conf = mox.Mox().CreateMockAnything() conf = mox.Mox().CreateMockAnything()
conf.metering_storage_engine = 'log' conf.database_connection = 'log://localhost'
engine = storage.get_engine(conf) engine = storage.get_engine(conf)
assert isinstance(engine, impl_log.LogStorage) assert isinstance(engine, impl_log.LogStorage)
def test_get_engine_no_such_engine(): def test_get_engine_no_such_engine():
conf = mox.Mox().CreateMockAnything() conf = mox.Mox().CreateMockAnything()
conf.metering_storage_engine = 'no-such-engine' conf.database_connection = 'no-such-engine://localhost'
try: try:
storage.get_engine(conf) storage.get_engine(conf)
except RuntimeError as err: except RuntimeError as err:

View File

@ -89,13 +89,10 @@ class MongoDBEngineTestBase(unittest.TestCase):
super(MongoDBEngineTestBase, self).setUp() super(MongoDBEngineTestBase, self).setUp()
self.conf = mox.Mox().CreateMockAnything() self.conf = mox.Mox().CreateMockAnything()
self.conf.metering_storage_engine = 'mongodb' self.conf.database_connection = 'mongodb://localhost/testdb'
self.conf.mongodb_host = 'localhost'
self.conf.mongodb_port = 27017
self.conf.mongodb_dbname = 'testdb'
self.conn = Connection(self.conf) self.conn = Connection(self.conf)
self.conn.conn.drop_database(self.conf.mongodb_dbname) self.conn.conn.drop_database('testdb')
self.db = self.conn.conn[self.conf.mongodb_dbname] self.db = self.conn.conn['testdb']
self.conn.db = self.db self.conn.db = self.db
self.counter = counter.Counter( self.counter = counter.Counter(