Add class 'StatsMixin'

1. Add class 'StatsMixin'
2. Fix the listener's stats

Change-Id: I7930d52c083c7089382583d657a77ac9969b46ff
Implements: blueprint stats-support
This commit is contained in:
chen-li 2016-08-22 22:24:26 -05:00
parent fe267c46b3
commit 8ee4def2b3
7 changed files with 206 additions and 45 deletions

View File

@ -30,6 +30,7 @@ LOG = logging.getLogger(__name__)
class BaseController(rest.RestController): class BaseController(rest.RestController):
def __init__(self): def __init__(self):
super(BaseController, self).__init__()
self.repositories = repositories.Repositories() self.repositories = repositories.Repositories()
self.handler = stevedore_driver.DriverManager( self.handler = stevedore_driver.DriverManager(
namespace='octavia.api.handlers', namespace='octavia.api.handlers',

View File

@ -12,8 +12,6 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
import logging
import pecan import pecan
from wsme import types as wtypes from wsme import types as wtypes
from wsmeext import pecan as wsme_pecan from wsmeext import pecan as wsme_pecan
@ -21,32 +19,16 @@ from wsmeext import pecan as wsme_pecan
from octavia.api.v1.controllers import base from octavia.api.v1.controllers import base
from octavia.api.v1.types import listener_statistics as ls_types from octavia.api.v1.types import listener_statistics as ls_types
from octavia.common import constants from octavia.common import constants
from octavia.common import data_models from octavia.common import stats
from octavia.common import exceptions
from octavia.i18n import _LI
LOG = logging.getLogger(__name__) class ListenerStatisticsController(base.BaseController,
stats.StatsMixin):
class ListenerStatisticsController(base.BaseController):
def __init__(self, listener_id): def __init__(self, listener_id):
super(ListenerStatisticsController, self).__init__() super(ListenerStatisticsController, self).__init__()
self.listener_id = listener_id self.listener_id = listener_id
def _get_db_ls(self, session):
"""Gets the current listener statistics object from the database."""
db_ls = self.repositories.listener_stats.get(
session, listener_id=self.listener_id)
if not db_ls:
LOG.info(_LI("Listener Statistics for Listener %s was not found"),
self.listener_id)
raise exceptions.NotFound(
resource=data_models.ListenerStatistics._name(),
id=self.listener_id)
return db_ls
@wsme_pecan.wsexpose({wtypes.text: ls_types.ListenerStatisticsResponse}) @wsme_pecan.wsexpose({wtypes.text: ls_types.ListenerStatisticsResponse})
def get_all(self): def get_all(self):
"""Gets a single listener's statistics details.""" """Gets a single listener's statistics details."""
@ -54,6 +36,7 @@ class ListenerStatisticsController(base.BaseController):
# listener statistics we are using the get_all method to only get # listener statistics we are using the get_all method to only get
# the single set of stats # the single set of stats
context = pecan.request.context.get('octavia_context') context = pecan.request.context.get('octavia_context')
db_ls = self._get_db_ls(context.session) data_stats = self.get_listener_stats(
context.session, self.listener_id)
return {constants.LISTENER: self._convert_db_to_type( return {constants.LISTENER: self._convert_db_to_type(
db_ls, ls_types.ListenerStatisticsResponse)} data_stats, ls_types.ListenerStatisticsResponse)}

View File

@ -146,6 +146,26 @@ class ListenerStatistics(BaseDataModel):
self.total_connections = total_connections self.total_connections = total_connections
self.request_errors = request_errors self.request_errors = request_errors
def get_stats(self):
stats = {
'bytes_in': self.bytes_in,
'bytes_out': self.bytes_out,
'active_connections': self.active_connections,
'total_connections': self.total_connections,
'request_errors': self.request_errors,
}
return stats
def __iadd__(self, other):
if isinstance(other, ListenerStatistics):
self.bytes_in += other.bytes_in
self.bytes_out += other.bytes_out
self.request_errors += other.request_errors
self.total_connections += other.total_connections
return self
class HealthMonitor(BaseDataModel): class HealthMonitor(BaseDataModel):

50
octavia/common/stats.py Normal file
View File

@ -0,0 +1,50 @@
# Copyright 2016 IBM
#
# 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.
import logging
from octavia.common import constants
from octavia.common import data_models
from octavia.db import repositories as repo
from octavia.i18n import _LW
LOG = logging.getLogger(__name__)
class StatsMixin(object):
def __init__(self):
super(StatsMixin, self).__init__()
self.listener_stats_repo = repo.ListenerStatisticsRepository()
self.repo_amphora = repo.AmphoraRepository()
def get_listener_stats(self, session, listener_id):
"""Gets the listener statistics data_models object."""
db_ls = self.listener_stats_repo.get_all(
session, listener_id=listener_id)
if not db_ls:
LOG.warning(
_LW("Listener Statistics for Listener %s was not found"),
listener_id)
statistics = data_models.ListenerStatistics(listener_id=listener_id)
for db_l in db_ls:
statistics += db_l
amp = self.repo_amphora.get(session, id=db_l.amphora_id)
if amp and amp.status == constants.AMPHORA_ALLOCATED:
statistics.active_connections += db_l.active_connections
return statistics

View File

@ -21,6 +21,7 @@ import sqlalchemy
from stevedore import driver as stevedore_driver from stevedore import driver as stevedore_driver
from octavia.common import constants from octavia.common import constants
from octavia.common import stats
from octavia.controller.healthmanager import update_serializer from octavia.controller.healthmanager import update_serializer
from octavia.controller.queue import event_queue from octavia.controller.queue import event_queue
from octavia.db import api as db_api from octavia.db import api as db_api
@ -211,11 +212,10 @@ class UpdateHealthDb(object):
LOG.error(_LE("Load balancer %s is not in DB"), lb_id) LOG.error(_LE("Load balancer %s is not in DB"), lb_id)
class UpdateStatsDb(object): class UpdateStatsDb(stats.StatsMixin):
def __init__(self): def __init__(self):
super(UpdateStatsDb, self).__init__() super(UpdateStatsDb, self).__init__()
self.listener_stats_repo = repo.ListenerStatisticsRepository()
self.event_streamer = event_queue.EventStreamerNeutron() self.event_streamer = event_queue.EventStreamerNeutron()
def emit(self, info_type, info_id, info_obj): def emit(self, info_type, info_id, info_obj):
@ -268,4 +268,7 @@ class UpdateStatsDb(object):
listener_id, amphora_id, stats) listener_id, amphora_id, stats)
self.listener_stats_repo.replace( self.listener_stats_repo.replace(
session, listener_id, amphora_id, **stats) session, listener_id, amphora_id, **stats)
self.emit('listener_stats', listener_id, stats)
listener_stats = self.get_listener_stats(session, listener_id)
self.emit(
'listener_stats', listener_id, listener_stats.get_stats())

View File

@ -0,0 +1,95 @@
# Copyright 2016 IBM
#
# 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.
import random
import mock
from oslo_utils import uuidutils
from octavia.common import constants
from octavia.common import data_models
from octavia.common import stats
from octavia.tests.unit import base
class TestStatsMixin(base.TestCase):
def setUp(self):
super(TestStatsMixin, self).setUp()
self.sm = stats.StatsMixin()
self.session = mock.MagicMock()
self.listener_id = uuidutils.generate_uuid()
self.amphora_id = uuidutils.generate_uuid()
self.repo_listener_stats = mock.MagicMock()
self.sm.listener_stats_repo = self.repo_listener_stats
self.fake_stats = data_models.ListenerStatistics(
listener_id=self.listener_id,
amphora_id=self.amphora_id,
bytes_in=random.randrange(1000000000),
bytes_out=random.randrange(1000000000),
active_connections=random.randrange(1000000000),
total_connections=random.randrange(1000000000),
request_errors=random.randrange(1000000000))
self.sm.listener_stats_repo.get_all.return_value = [self.fake_stats]
self.repo_amphora = mock.MagicMock()
self.sm.repo_amphora = self.repo_amphora
def test_get_listener_stats(self):
fake_amp = mock.MagicMock()
fake_amp.status = constants.AMPHORA_ALLOCATED
self.sm.repo_amphora.get.return_value = fake_amp
ls_stats = self.sm.get_listener_stats(
self.session, self.listener_id)
self.repo_listener_stats.get_all.assert_called_once_with(
self.session, listener_id=self.listener_id)
self.repo_amphora.get.assert_called_once_with(
self.session, id=self.amphora_id)
self.assertEqual(self.fake_stats.bytes_in, ls_stats.bytes_in)
self.assertEqual(self.fake_stats.bytes_out, ls_stats.bytes_out)
self.assertEqual(
self.fake_stats.active_connections, ls_stats.active_connections)
self.assertEqual(
self.fake_stats.total_connections, ls_stats.total_connections)
self.assertEqual(
self.fake_stats.request_errors, ls_stats.request_errors)
self.assertEqual(self.listener_id, ls_stats.listener_id)
self.assertIsNone(ls_stats.amphora_id)
def test_get_listener_stats_with_amphora_deleted(self):
fake_amp = mock.MagicMock()
fake_amp.status = constants.DELETED
self.sm.repo_amphora.get.return_value = fake_amp
ls_stats = self.sm.get_listener_stats(self.session, self.listener_id)
self.repo_listener_stats.get_all.assert_called_once_with(
self.session, listener_id=self.listener_id)
self.repo_amphora.get.assert_called_once_with(
self.session, id=self.amphora_id)
self.assertEqual(self.fake_stats.bytes_in, ls_stats.bytes_in)
self.assertEqual(self.fake_stats.bytes_out, ls_stats.bytes_out)
self.assertEqual(0, ls_stats.active_connections)
self.assertEqual(
self.fake_stats.total_connections, ls_stats.total_connections)
self.assertEqual(
self.fake_stats.request_errors, ls_stats.request_errors)
self.assertEqual(self.listener_id, ls_stats.listener_id)
self.assertIsNone(ls_stats.amphora_id)

View File

@ -457,14 +457,20 @@ class TestUpdateStatsDb(base.TestCase):
self.listener_stats_repo = mock.MagicMock() self.listener_stats_repo = mock.MagicMock()
self.sm.listener_stats_repo = self.listener_stats_repo self.sm.listener_stats_repo = self.listener_stats_repo
self.bytes_in = random.randrange(1000000000)
self.bytes_out = random.randrange(1000000000)
self.request_errors = random.randrange(1000000000)
self.active_conns = random.randrange(1000000000)
self.total_conns = random.randrange(1000000000)
self.loadbalancer_id = uuidutils.generate_uuid() self.loadbalancer_id = uuidutils.generate_uuid()
self.listener_id = uuidutils.generate_uuid() self.listener_id = uuidutils.generate_uuid()
self.listener_stats = data_models.ListenerStatistics(
listener_id=self.listener_id,
bytes_in=random.randrange(1000000000),
bytes_out=random.randrange(1000000000),
active_connections=random.randrange(1000000000),
total_connections=random.randrange(1000000000),
request_errors=random.randrange(1000000000))
self.sm.get_listener_stats = mock.MagicMock(
return_value=self.listener_stats)
@mock.patch('octavia.db.api.get_session') @mock.patch('octavia.db.api.get_session')
def test_update_stats(self, session): def test_update_stats(self, session):
@ -474,11 +480,11 @@ class TestUpdateStatsDb(base.TestCase):
self.listener_id: { self.listener_id: {
"status": constants.OPEN, "status": constants.OPEN,
"stats": { "stats": {
"ereq": self.request_errors, "ereq": self.listener_stats.request_errors,
"conns": self.active_conns, "conns": self.listener_stats.active_connections,
"totconns": self.total_conns, "totconns": self.listener_stats.total_connections,
"rx": self.bytes_in, "rx": self.listener_stats.bytes_in,
"tx": self.bytes_out, "tx": self.listener_stats.bytes_out,
}, },
"pools": { "pools": {
"pool-id-1": { "pool-id-1": {
@ -496,17 +502,20 @@ class TestUpdateStatsDb(base.TestCase):
self.listener_stats_repo.replace.assert_called_once_with( self.listener_stats_repo.replace.assert_called_once_with(
'blah', self.listener_id, self.loadbalancer_id, 'blah', self.listener_id, self.loadbalancer_id,
bytes_in=self.bytes_in, bytes_out=self.bytes_out, bytes_in=self.listener_stats.bytes_in,
active_connections=self.active_conns, bytes_out=self.listener_stats.bytes_out,
total_connections=self.total_conns, active_connections=self.listener_stats.active_connections,
request_errors=self.request_errors) total_connections=self.listener_stats.total_connections,
request_errors=self.listener_stats.request_errors)
self.event_client.cast.assert_called_once_with( self.event_client.cast.assert_called_once_with(
{}, 'update_info', container={ {}, 'update_info', container={
'info_type': 'listener_stats', 'info_type': 'listener_stats',
'info_id': self.listener_id, 'info_id': self.listener_id,
'info_payload': { 'info_payload': {
'bytes_in': self.bytes_in, 'bytes_in': self.listener_stats.bytes_in,
'total_connections': self.total_conns, 'total_connections':
'active_connections': self.active_conns, self.listener_stats.total_connections,
'bytes_out': self.bytes_out, 'active_connections':
'request_errors': self.request_errors}}) self.listener_stats.active_connections,
'bytes_out': self.listener_stats.bytes_out,
'request_errors': self.listener_stats.request_errors}})