Added multi-tenancy support

Change-Id: I89e75d4dd36410ab7e24f81261ad8703dab11297
This commit is contained in:
Stéphane Albert 2014-11-25 12:14:00 +01:00
parent 32d83e3614
commit 1f67217cde
16 changed files with 408 additions and 169 deletions

View File

@ -15,6 +15,8 @@
# #
# @author: Stéphane Albert # @author: Stéphane Albert
# #
import datetime
from oslo.config import cfg from oslo.config import cfg
import pecan import pecan
from pecan import rest from pecan import rest
@ -242,19 +244,34 @@ class ReportController(rest.RestController):
""" """
_custom_actions = { _custom_actions = {
'total': ['GET'] 'total': ['GET'],
'tenants': ['GET']
} }
@wsme_pecan.wsexpose(float) @wsme_pecan.wsexpose([wtypes.text],
def total(self): datetime.datetime,
"""Return the amount to pay for the current month. datetime.datetime)
def tenants(self, begin=None, end=None):
"""Return the list of rated tenants.
"""
storage = pecan.request.storage_backend
tenants = storage.get_tenants(begin, end)
return tenants
@wsme_pecan.wsexpose(float,
datetime.datetime,
datetime.datetime,
wtypes.text)
def total(self, begin=None, end=None, tenant_id=None):
"""Return the amount to pay for a given period.
""" """
storage = pecan.request.storage_backend storage = pecan.request.storage_backend
# FIXME(sheeprine): We should filter on user id. # FIXME(sheeprine): We should filter on user id.
# Use keystone token information by default but make it overridable and # Use keystone token information by default but make it overridable and
# enforce it by policy engine # enforce it by policy engine
total = storage.get_total() total = storage.get_total(begin, end, tenant_id)
return total return total

View File

@ -131,10 +131,11 @@ class BillingController(rest.RestController):
""" """
config = BillingConfigController()
enabled = BillingEnableController()
def __init__(self): def __init__(self):
if not hasattr(self, 'config'):
self.config = BillingConfigController()
if not hasattr(self, 'enabled'):
self.enabled = BillingEnableController()
if hasattr(self, 'module_name'): if hasattr(self, 'module_name'):
self.config.module_name = self.module_name self.config.module_name = self.module_name
self.enabled.module_name = self.module_name self.enabled.module_name = self.module_name
@ -159,8 +160,8 @@ class BillingProcessorBase(object):
controller = BillingController controller = BillingController
def __init__(self): def __init__(self, tenant_id=None):
pass self._tenant_id = tenant_id
@abc.abstractproperty @abc.abstractproperty
def enabled(self): def enabled(self):

View File

@ -227,7 +227,8 @@ class BasicHashMap(billing.BillingProcessorBase):
controller = BasicHashMapController controller = BasicHashMapController
db_api = api.get_instance() db_api = api.get_instance()
def __init__(self): def __init__(self, tenant_id=None):
super(BasicHashMap, self).__init__(tenant_id)
self._billing_info = {} self._billing_info = {}
self._load_billing_rates() self._load_billing_rates()

View File

@ -37,8 +37,8 @@ class Noop(billing.BillingProcessorBase):
controller = NoopController controller = NoopController
def __init__(self): def __init__(self, tenant_id=None):
pass super(Noop, self).__init__(tenant_id)
@property @property
def enabled(self): def enabled(self):

View File

@ -15,6 +15,8 @@
# #
# @author: Stéphane Albert # @author: Stéphane Albert
# #
from __future__ import print_function
from oslo.config import cfg from oslo.config import cfg
from stevedore import driver from stevedore import driver
@ -44,14 +46,75 @@ def load_output_backend():
return backend return backend
def main(): class DBCommand(object):
service.prepare_service()
output_backend = load_output_backend()
storage_backend = load_storage_backend()
wo = write_orchestrator.WriteOrchestrator(output_backend, def __init__(self):
'writer', self._storage = None
storage_backend) self._output = None
self._load_storage_backend()
self._load_output_backend()
def _load_storage_backend(self):
storage_args = {'period': CONF.collect.period}
CONF.import_opt('backend', 'cloudkitty.storage', 'storage')
backend = driver.DriverManager(
STORAGES_NAMESPACE,
CONF.storage.backend,
invoke_on_load=True,
invoke_kwds=storage_args).driver
self._storage = backend
def _load_output_backend(self):
CONF.import_opt('backend', 'cloudkitty.config', 'output')
backend = i_utils.import_class(CONF.output.backend)
self._output = backend
def generate(self):
if not CONF.command.tenant:
tenants = self._storage.get_tenants(CONF.command.begin,
CONF.command.end)
else:
tenants = [CONF.command.tenant]
for tenant in tenants:
wo = write_orchestrator.WriteOrchestrator(self._output,
tenant,
self._storage)
wo.init_writing_pipeline() wo.init_writing_pipeline()
if not CONF.command.begin:
wo.restart_month() wo.restart_month()
wo.process() wo.process()
def tenants_list(self):
tenants = self._storage.get_tenants(CONF.command.begin,
CONF.command.end)
print('Tenant list:')
for tenant in tenants:
print(tenant)
def add_command_parsers(subparsers):
command_object = DBCommand()
parser = subparsers.add_parser('generate')
parser.set_defaults(func=command_object.generate)
parser.add_argument('--tenant', nargs='?')
parser.add_argument('--begin', nargs='?')
parser.add_argument('--end', nargs='?')
parser = subparsers.add_parser('tenants_list')
parser.set_defaults(func=command_object.tenants_list)
parser.add_argument('--begin', nargs='?')
parser.add_argument('--end', nargs='?')
command_opt = cfg.SubCommandOpt('command',
title='Command',
help='Available commands',
handler=add_command_parsers)
CONF.register_cli_opt(command_opt)
def main():
service.prepare_service()
CONF.command.func()

View File

@ -94,7 +94,8 @@ class CeilometerCollector(collector.BaseCollector):
self._cacher = CeilometerResourceCacher() self._cacher = CeilometerResourceCacher()
self._conn = cclient.get_client('2', os_username=self.user, self._conn = cclient.get_client('2',
os_username=self.user,
os_password=self.password, os_password=self.password,
os_auth_url=self.keystone_url, os_auth_url=self.keystone_url,
os_tenant_name=self.tenant, os_tenant_name=self.tenant,

View File

@ -24,7 +24,7 @@ _FACADE = None
def _create_facade_lazily(): def _create_facade_lazily():
global _FACADE global _FACADE
if _FACADE is None: if _FACADE is None:
_FACADE = session.EngineFacade.from_config(cfg.CONF) _FACADE = session.EngineFacade.from_config(cfg.CONF, sqlite_fk=True)
return _FACADE return _FACADE

View File

@ -16,6 +16,8 @@
# #
# @author: Stéphane Albert # @author: Stéphane Albert
# #
import decimal
import eventlet import eventlet
from keystoneclient.v2_0 import client as kclient from keystoneclient.v2_0 import client as kclient
from oslo.config import cfg from oslo.config import cfg
@ -67,7 +69,8 @@ class BillingEndpoint(object):
def quote(self, ctxt, res_data): def quote(self, ctxt, res_data):
LOG.debug('Received quote from RPC.') LOG.debug('Received quote from RPC.')
return self._orchestrator.process_quote(res_data) worker = APIWorker()
return worker.quote(res_data)
def reload_module(self, ctxt, name): def reload_module(self, ctxt, name):
LOG.info('Received reload command for module {}.'.format(name)) LOG.info('Received reload command for module {}.'.format(name))
@ -91,13 +94,112 @@ class BillingEndpoint(object):
self._pending_reload.remove(name) self._pending_reload.remove(name)
class BaseWorker(object):
def __init__(self, tenant_id=None):
self._tenant_id = tenant_id
# Billing processors
self._processors = {}
self._load_billing_processors()
def _load_billing_processors(self):
self._processors = {}
processors = extension_manager.EnabledExtensionManager(
PROCESSORS_NAMESPACE,
invoke_kwds={'tenant_id': self._tenant_id}
)
for processor in processors:
b_name = processor.name
b_obj = processor.obj
self._processors[b_name] = b_obj
class APIWorker(BaseWorker):
def __init__(self, tenant_id=None):
super(APIWorker, self).__init__(tenant_id)
def quote(self, res_data):
for processor in self._processors.values():
processor.process(res_data)
price = decimal.Decimal(0)
for res in res_data:
for res_usage in res['usage'].values():
for data in res_usage:
price += data.get('billing', {}).get('price', 0.0)
return price
class Worker(BaseWorker):
def __init__(self, collector, storage, tenant_id=None):
self._collector = collector
self._storage = storage
self._period = CONF.collect.period
self._wait_time = CONF.collect.wait_periods * CONF.collect.period
super(Worker, self).__init__(tenant_id)
def _collect(self, service, start_timestamp):
next_timestamp = start_timestamp + CONF.collect.period
raw_data = self._collector.retrieve(service,
start_timestamp,
next_timestamp,
self._tenant_id)
timed_data = [{'period': {'begin': start_timestamp,
'end': next_timestamp},
'usage': raw_data}]
return timed_data
def check_state(self):
timestamp = self._storage.get_state(self._tenant_id)
if not timestamp:
month_start = ck_utils.get_month_start()
return ck_utils.dt2ts(month_start)
now = ck_utils.utcnow_ts()
next_timestamp = timestamp + self._period
if next_timestamp + self._wait_time < now:
return next_timestamp
return 0
def run(self):
while True:
timestamp = self.check_state()
if not timestamp:
break
for service in CONF.collect.services:
data = self._collect(service, timestamp)
# Billing
for processor in self._processors.values():
processor.process(data)
# Writing
self._storage.append(data, self._tenant_id)
# We're getting a full period so we directly commit
self._storage.commit(self._tenant_id)
class Orchestrator(object): class Orchestrator(object):
def __init__(self): def __init__(self):
self.admin_ks = kclient.Client(username=CONF.auth.username, # Load credentials informations
password=CONF.auth.password, self.user = CONF.auth.username
tenant_name=CONF.auth.tenant, self.password = CONF.auth.password
region_name=CONF.auth.region, self.tenant = CONF.auth.tenant
auth_url=CONF.auth.url) self.region = CONF.auth.region
self.keystone_url = CONF.auth.url
# Initialize keystone admin session
self.admin_ks = kclient.Client(username=self.user,
password=self.password,
tenant_name=self.tenant,
region_name=self.region,
auth_url=self.keystone_url)
# Transformers # Transformers
self.transformers = {} self.transformers = {}
@ -119,15 +221,25 @@ class Orchestrator(object):
invoke_on_load=True, invoke_on_load=True,
invoke_kwds=storage_args).driver invoke_kwds=storage_args).driver
# Billing processors
self.b_processors = {}
self._load_billing_processors()
# RPC # RPC
self.server = None self.server = None
self._billing_endpoint = BillingEndpoint(self) self._billing_endpoint = BillingEndpoint(self)
self._init_messaging() self._init_messaging()
def _load_tenant_list(self):
ks = kclient.Client(username=self.user,
password=self.password,
auth_url=self.keystone_url,
region_name=self.region)
tenant_list = ks.tenants.list()
self._tenants = []
for tenant in tenant_list:
roles = self.admin_ks.roles.roles_for_user(self.admin_ks.user_id,
tenant)
for role in roles:
if role.name == 'rating':
self._tenants.append(tenant)
def _init_messaging(self): def _init_messaging(self):
target = messaging.Target(topic='cloudkitty', target = messaging.Target(topic='cloudkitty',
server=CONF.host, server=CONF.host,
@ -138,8 +250,8 @@ class Orchestrator(object):
self.server = rpc.get_server(target, endpoints) self.server = rpc.get_server(target, endpoints)
self.server.start() self.server.start()
def _check_state(self): def _check_state(self, tenant_id):
timestamp = self.storage.get_state() timestamp = self.storage.get_state(tenant_id)
if not timestamp: if not timestamp:
month_start = ck_utils.get_month_start() month_start = ck_utils.get_month_start()
return ck_utils.dt2ts(month_start) return ck_utils.dt2ts(month_start)
@ -173,72 +285,28 @@ class Orchestrator(object):
t_obj = transformer.obj t_obj = transformer.obj
self.transformers[t_name] = t_obj self.transformers[t_name] = t_obj
def _load_billing_processors(self):
self.b_processors = {}
processors = extension_manager.EnabledExtensionManager(
PROCESSORS_NAMESPACE,
)
for processor in processors:
b_name = processor.name
b_obj = processor.obj
self.b_processors[b_name] = b_obj
def process_quote(self, res_data):
for processor in self.b_processors.values():
processor.process(res_data)
price = 0.0
for res in res_data:
for res_usage in res['usage'].values():
for data in res_usage:
price += data.get('billing', {}).get('price', 0.0)
return price
def process_messages(self): def process_messages(self):
pending_reload = self._billing_endpoint.get_reload_list() # TODO(sheeprine): Code kept to handle threading and asynchronous
pending_states = self._billing_endpoint.get_module_state() # reloading
for name in pending_reload: # pending_reload = self._billing_endpoint.get_reload_list()
if name in self.b_processors: # pending_states = self._billing_endpoint.get_module_state()
if name in self.b_processors.keys(): pass
LOG.info('Reloading configuration of {} module.'.format(
name))
self.b_processors[name].reload_config()
else:
LOG.info('Tried to reload a disabled module: {}.'.format(
name))
for name, status in pending_states.items():
if name in self.b_processors and not status:
LOG.info('Disabling {} module.'.format(name))
self.b_processors.pop(name)
else:
LOG.info('Enabling {} module.'.format(name))
processors = extension_manager.EnabledExtensionManager(
PROCESSORS_NAMESPACE)
for processor in processors:
if processor.name == name:
self.b_processors[name] = processor
def process(self): def process(self):
while True: while True:
self.process_messages() self.process_messages()
timestamp = self._check_state() self._load_tenant_list()
if not timestamp: while len(self._tenants):
for tenant in self._tenants:
if not self._check_state(tenant.id):
self._tenants.remove(tenant)
else:
worker = Worker(self.collector,
self.storage,
tenant.id)
worker.run()
# FIXME(sheeprine): We may cause a drift here
eventlet.sleep(CONF.collect.period) eventlet.sleep(CONF.collect.period)
continue
for service in CONF.collect.services:
data = self._collect(service, timestamp)
# Billing
for processor in self.b_processors.values():
processor.process(data)
# Writing
self.storage.append(data)
# We're getting a full period so we directly commit
self.storage.commit()
def terminate(self): def terminate(self):
pass pass

View File

@ -62,10 +62,10 @@ class BaseStorage(object):
self._period = period self._period = period
# State vars # State vars
self.usage_start = None self.usage_start = {}
self.usage_start_dt = None self.usage_start_dt = {}
self.usage_end = None self.usage_end = {}
self.usage_end_dt = None self.usage_end_dt = {}
@staticmethod @staticmethod
def init(): def init():
@ -93,38 +93,45 @@ class BaseStorage(object):
if candidate_ts: if candidate_ts:
return candidate_ts, json_data.pop(candidate_idx)['usage'] return candidate_ts, json_data.pop(candidate_idx)['usage']
def _pre_commit(self): def _pre_commit(self, tenant_id):
"""Called before every commit. """Called before every commit.
""" """
@abc.abstractmethod @abc.abstractmethod
def _commit(self): def _commit(self, tenant_id):
"""Push data to the storage backend. """Push data to the storage backend.
""" """
def _post_commit(self): def _post_commit(self, tenant_id):
"""Called after every commit. """Called after every commit.
""" """
@abc.abstractmethod @abc.abstractmethod
def _dispatch(self, data): def _dispatch(self, data, tenant_id):
"""Process rated data. """Process rated data.
:param data: The rated data frames. :param data: The rated data frames.
""" """
@abc.abstractmethod @abc.abstractmethod
def get_state(self): def get_state(self, tenant_id=None):
"""Return the last written frame's timestamp. """Return the last written frame's timestamp.
:param tenant_id: Tenant ID to filter on.
"""
@abc.abstractmethod
def get_total(self, tenant_id=None):
"""Return the current total.
""" """
@abc.abstractmethod @abc.abstractmethod
def get_total(self): def get_tenants(self, begin=None, end=None):
"""Return the current total. """Return the list of rated tenants.
""" """
@ -138,31 +145,37 @@ class BaseStorage(object):
:type end: datetime.datetime :type end: datetime.datetime
:param res_type: (Optional) Filter on the resource type. :param res_type: (Optional) Filter on the resource type.
:type res_type: str :type res_type: str
:param tenant_id: (Optional) Filter on the tenant_id.
:type res_type: str
""" """
def append(self, raw_data): def append(self, raw_data, tenant_id):
"""Append rated data before committing them to the backend. """Append rated data before committing them to the backend.
:param raw_data: The rated data frames. :param raw_data: The rated data frames.
:param tenant_id: Tenant the frame is belonging.
""" """
while raw_data: while raw_data:
usage_start, data = self._filter_period(raw_data) usage_start, data = self._filter_period(raw_data)
if self.usage_end is not None and usage_start >= self.usage_end: usage_end = self.usage_end.get(tenant_id)
self.commit() if usage_end is not None and usage_start >= usage_end:
self.usage_start = None self.commit(tenant_id)
self.usage_start.pop(tenant_id)
if self.usage_start is None: if self.usage_start.get(tenant_id) is None:
self.usage_start = usage_start self.usage_start[tenant_id] = usage_start
self.usage_end = usage_start + self._period self.usage_end[tenant_id] = usage_start + self._period
self.usage_start_dt = ck_utils.ts2dt(self.usage_start) self.usage_start_dt[tenant_id] = ck_utils.ts2dt(
self.usage_end_dt = ck_utils.ts2dt(self.usage_end) self.usage_start.get(tenant_id))
self.usage_end_dt[tenant_id] = ck_utils.ts2dt(
self.usage_end.get(tenant_id))
self._dispatch(data) self._dispatch(data, tenant_id)
def commit(self): def commit(self, tenant_id):
"""Commit the changes to the backend. """Commit the changes to the backend.
""" """
self._pre_commit() self._pre_commit(tenant_id)
self._commit() self._commit(tenant_id)
self._post_commit() self._post_commit(tenant_id)

View File

@ -33,69 +33,97 @@ class SQLAlchemyStorage(storage.BaseStorage):
""" """
def __init__(self, period=3600): def __init__(self, period=3600):
super(SQLAlchemyStorage, self).__init__(period) super(SQLAlchemyStorage, self).__init__(period)
self._session = None self._session = {}
@staticmethod @staticmethod
def init(): def init():
migration.upgrade('head') migration.upgrade('head')
def _commit(self): def _commit(self, tenant_id):
self._session.commit() self._session[tenant_id].commit()
self._session.begin() self._session[tenant_id].begin()
def _dispatch(self, data): def _dispatch(self, data, tenant_id):
for service in data: for service in data:
for frame in data[service]: for frame in data[service]:
self._append_time_frame(service, frame) self._append_time_frame(service, frame, tenant_id)
# HACK(adriant) Quick hack to allow billing windows to # HACK(adriant) Quick hack to allow billing windows to
# progress. This check/insert probably ought to be moved # progress. This check/insert probably ought to be moved
# somewhere else. # somewhere else.
if not data[service]: if not data[service]:
empty_frame = {'vol': {'qty': 0, 'unit': 'None'}, empty_frame = {'vol': {'qty': 0, 'unit': 'None'},
'billing': {'price': 0}, 'desc': ''} 'billing': {'price': 0}, 'desc': ''}
self._append_time_frame(service, empty_frame) self._append_time_frame(service, empty_frame, tenant_id)
def append(self, raw_data): def append(self, raw_data, tenant_id):
if not self._session: session = self._session.get(tenant_id)
self._session = db.get_session() if not session:
self._session.begin() self._session[tenant_id] = db.get_session()
super(SQLAlchemyStorage, self).append(raw_data) self._session[tenant_id].begin()
super(SQLAlchemyStorage, self).append(raw_data, tenant_id)
def get_state(self): def get_state(self, tenant_id=None):
session = db.get_session() session = db.get_session()
r = utils.model_query( q = utils.model_query(
models.RatedDataFrame, models.RatedDataFrame,
session session
).order_by( )
if tenant_id:
q = q.filter(
models.RatedDataFrame.tenant_id == tenant_id
)
r = q.order_by(
models.RatedDataFrame.begin.desc() models.RatedDataFrame.begin.desc()
).first() ).first()
if r: if r:
return ck_utils.dt2ts(r.begin) return ck_utils.dt2ts(r.begin)
def get_total(self): def get_total(self, begin=None, end=None, tenant_id=None):
model = models.RatedDataFrame model = models.RatedDataFrame
# Boundary calculation # Boundary calculation
month_start = ck_utils.get_month_start() if not begin:
month_end = ck_utils.get_next_month() begin = ck_utils.get_month_start()
if not end:
end = ck_utils.get_next_month()
session = db.get_session() session = db.get_session()
rate = session.query( q = session.query(
sqlalchemy.func.sum(model.rate).label('rate') sqlalchemy.func.sum(model.rate).label('rate')
).filter( )
model.begin >= month_start, if tenant_id:
model.end <= month_end q = q.filter(
models.RatedDataFrame.tenant_id == tenant_id
)
rate = q.filter(
model.begin >= begin,
model.end <= end
).scalar() ).scalar()
return rate return rate
def get_time_frame(self, begin, end, **filters): def get_tenants(self, begin=None, end=None):
"""Return a list of time frames. model = models.RatedDataFrame
:param start: Filter from `start`. # Boundary calculation
:param end: Filter to `end`. if not begin:
:param unit: Filter on an unit type. begin = ck_utils.get_month_start()
:param res_type: Filter on a resource type. if not end:
""" end = ck_utils.get_next_month()
session = db.get_session()
q = utils.model_query(
model,
session
).filter(
model.begin >= begin,
model.end <= end
)
tenants = q.distinct().values(
model.tenant_id
)
return [tenant.tenant_id for tenant in tenants]
def get_time_frame(self, begin, end, **filters):
model = models.RatedDataFrame model = models.RatedDataFrame
session = db.get_session() session = db.get_session()
q = utils.model_query( q = utils.model_query(
@ -112,30 +140,33 @@ class SQLAlchemyStorage(storage.BaseStorage):
raise storage.NoTimeFrame() raise storage.NoTimeFrame()
return [entry.to_cloudkitty() for entry in r] return [entry.to_cloudkitty() for entry in r]
def _append_time_frame(self, res_type, frame): def _append_time_frame(self, res_type, frame, tenant_id):
vol_dict = frame['vol'] vol_dict = frame['vol']
qty = vol_dict['qty'] qty = vol_dict['qty']
unit = vol_dict['unit'] unit = vol_dict['unit']
rating_dict = frame['billing'] rating_dict = frame['billing']
rate = rating_dict['price'] rate = rating_dict['price']
desc = json.dumps(frame['desc']) desc = json.dumps(frame['desc'])
self.add_time_frame(self.usage_start_dt, self.add_time_frame(self.usage_start_dt.get(tenant_id),
self.usage_end_dt, self.usage_end_dt.get(tenant_id),
tenant_id,
unit, unit,
qty, qty,
res_type, res_type,
rate, rate,
desc) desc)
def add_time_frame(self, begin, end, unit, qty, res_type, rate, desc): def add_time_frame(self, begin, end, tenant_id, unit, qty, res_type,
rate, desc):
"""Create a new time frame. """Create a new time frame.
""" """
frame = models.RatedDataFrame(begin=begin, frame = models.RatedDataFrame(begin=begin,
end=end, end=end,
tenant_id=tenant_id,
unit=unit, unit=unit,
qty=qty, qty=qty,
res_type=res_type, res_type=res_type,
rate=rate, rate=rate,
desc=desc) desc=desc)
self._session.add(frame) self._session[tenant_id].add(frame)

View File

@ -0,0 +1,27 @@
"""added tenant informations
Revision ID: 792b438b663
Revises: 17fd1b237aa3
Create Date: 2014-12-02 13:12:11.328534
"""
# revision identifiers, used by Alembic.
revision = '792b438b663'
down_revision = '17fd1b237aa3'
from alembic import op
import sqlalchemy as sa
from cloudkitty.storage.sqlalchemy import models
def upgrade():
op.add_column('rated_data_frames',
sa.Column('tenant_id',
sa.String(length=32),
nullable=True))
def downgrade():
op.drop_column('rated_data_frames', 'tenant_id')

View File

@ -21,6 +21,8 @@ from oslo.db.sqlalchemy import models
import sqlalchemy import sqlalchemy
from sqlalchemy.ext import declarative from sqlalchemy.ext import declarative
from cloudkitty import utils as ck_utils
Base = declarative.declarative_base() Base = declarative.declarative_base()
@ -34,6 +36,8 @@ class RatedDataFrame(Base, models.ModelBase):
id = sqlalchemy.Column(sqlalchemy.Integer, id = sqlalchemy.Column(sqlalchemy.Integer,
primary_key=True) primary_key=True)
tenant_id = sqlalchemy.Column(sqlalchemy.String(32),
nullable=True)
begin = sqlalchemy.Column(sqlalchemy.DateTime, begin = sqlalchemy.Column(sqlalchemy.DateTime,
nullable=False) nullable=False)
end = sqlalchemy.Column(sqlalchemy.DateTime, end = sqlalchemy.Column(sqlalchemy.DateTime,
@ -50,20 +54,32 @@ class RatedDataFrame(Base, models.ModelBase):
nullable=False) nullable=False)
def to_cloudkitty(self): def to_cloudkitty(self):
period_dict = {} # Rating informations
period_dict['begin'] = self.begin.isoformat()
period_dict['end'] = self.end.isoformat()
rating_dict = {} rating_dict = {}
rating_dict['price'] = self.rate rating_dict['price'] = self.rate
# Volume informations
vol_dict = {} vol_dict = {}
vol_dict['qty'] = self.qty vol_dict['qty'] = self.qty
vol_dict['unit'] = self.unit vol_dict['unit'] = self.unit
res_dict = {} res_dict = {}
# Encapsulate informations in a resource dict
res_dict['billing'] = rating_dict res_dict['billing'] = rating_dict
res_dict['desc'] = json.loads(self.desc) res_dict['desc'] = json.loads(self.desc)
res_dict['vol'] = vol_dict res_dict['vol'] = vol_dict
res_dict['tenant_id'] = self.tenant_id
# Add resource to the usage dict
usage_dict = {} usage_dict = {}
usage_dict[self.res_type] = [res_dict] usage_dict[self.res_type] = [res_dict]
# Time informations
period_dict = {}
period_dict['begin'] = ck_utils.dt2iso(self.begin)
period_dict['end'] = ck_utils.dt2iso(self.end)
# Add period to the resource informations
ck_dict = {} ck_dict = {}
ck_dict['period'] = period_dict ck_dict['period'] = period_dict
ck_dict['usage'] = usage_dict ck_dict['usage'] = usage_dict

View File

@ -34,16 +34,16 @@ class WriteOrchestrator(object):
""" """
def __init__(self, def __init__(self,
backend, backend,
user_id, tenant_id,
storage, storage,
basepath=None, basepath=None,
period=3600): period=3600):
self._backend = backend self._backend = backend
self._uid = user_id self._tenant_id = tenant_id
self._storage = storage self._storage = storage
self._basepath = basepath self._basepath = basepath
self._period = period self._period = period
self._sm = state.DBStateManager(self._uid, self._sm = state.DBStateManager(self._tenant_id,
'writer_status') 'writer_status')
self._write_pipeline = [] self._write_pipeline = []
@ -64,7 +64,7 @@ class WriteOrchestrator(object):
def add_writer(self, writer_class): def add_writer(self, writer_class):
writer = writer_class(self, writer = writer_class(self,
self._uid, self._tenant_id,
self._backend, self._backend,
self._basepath) self._basepath)
self._write_pipeline.append(writer) self._write_pipeline.append(writer)
@ -97,7 +97,8 @@ class WriteOrchestrator(object):
timeframe_end = timeframe + self._period timeframe_end = timeframe + self._period
try: try:
data = self._storage.get_time_frame(timeframe, data = self._storage.get_time_frame(timeframe,
timeframe_end) timeframe_end,
tenant_id=self._tenant_id)
except storage.NoTimeFrame: except storage.NoTimeFrame:
return None return None
return data return data

View File

@ -28,11 +28,11 @@ class BaseReportWriter(object):
"""Base report writer.""" """Base report writer."""
report_type = None report_type = None
def __init__(self, write_orchestrator, user_id, backend, basepath=None): def __init__(self, write_orchestrator, tenant_id, backend, basepath=None):
self._write_orchestrator = write_orchestrator self._write_orchestrator = write_orchestrator
self._backend = backend self._backend = backend
self._uid = user_id self._tenant_id = tenant_id
self._sm = state.DBStateManager(self._uid, self._sm = state.DBStateManager(self._tenant_id,
self.report_type) self.report_type)
self._report = None self._report = None
self._period = 3600 self._period = 3600

View File

@ -37,7 +37,7 @@ class OSRFBackend(writer.BaseReportWriter):
report_type = 'osrf' report_type = 'osrf'
def _gen_filename(self, timeframe): def _gen_filename(self, timeframe):
filename = '{}-osrf-{}-{:02d}.json'.format(self._uid, filename = '{}-osrf-{}-{:02d}.json'.format(self._tenant_id,
timeframe.year, timeframe.year,
timeframe.month) timeframe.month)
if self._basepath: if self._basepath:

View File

@ -5,7 +5,7 @@ python-keystoneclient
iso8601 iso8601
PasteDeploy==1.5.2 PasteDeploy==1.5.2
posix_ipc posix_ipc
pecan==0.5.0 pecan>=0.8.0
WSME>=0.6,!=0.6.2 WSME>=0.6,!=0.6.2
oslo.config>=1.2.0 oslo.config>=1.2.0
oslo.messaging<1.6.0 oslo.messaging<1.6.0