Merge "Add a new API to list loadbalancer statistics"
This commit is contained in:
commit
ae0ca96e8a
@ -168,6 +168,42 @@ Retrieve details of a load balancer.
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
List Load Balancer Statistics
|
||||||
|
*****************************
|
||||||
|
|
||||||
|
Retrieve the stats of a load balancer.
|
||||||
|
|
||||||
|
+----------------+-----------------------------------------------------------+
|
||||||
|
| Request Type | ``GET`` |
|
||||||
|
+----------------+-----------------------------------------------------------+
|
||||||
|
| Endpoint | ``URL/v1/loadbalancers/{lb_id}/stats`` |
|
||||||
|
+----------------+---------+-------------------------------------------------+
|
||||||
|
| | Success | 200 |
|
||||||
|
| Response Codes +---------+-------------------------------------------------+
|
||||||
|
| | Error | 401, 404, 500 |
|
||||||
|
+----------------+---------+-------------------------------------------------+
|
||||||
|
|
||||||
|
**Response Example**::
|
||||||
|
|
||||||
|
{
|
||||||
|
"loadbalancer": {
|
||||||
|
"bytes_in": 0,
|
||||||
|
"bytes_out": 0,
|
||||||
|
"active_connections": 0,
|
||||||
|
"total_connections": 0,
|
||||||
|
"request_errors": 0,
|
||||||
|
"listeners": [{
|
||||||
|
"id": "uuid"
|
||||||
|
"bytes_in": 0,
|
||||||
|
"bytes_out": 0,
|
||||||
|
"active_connections": 0,
|
||||||
|
"total_connections": 0,
|
||||||
|
"request_errors": 0,
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
Create Load Balancer
|
Create Load Balancer
|
||||||
********************
|
********************
|
||||||
|
|
||||||
|
@ -22,6 +22,7 @@ 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.controllers import listener
|
from octavia.api.v1.controllers import listener
|
||||||
|
from octavia.api.v1.controllers import load_balancer_statistics as lb_stats
|
||||||
from octavia.api.v1.controllers import pool
|
from octavia.api.v1.controllers import pool
|
||||||
from octavia.api.v1.types import load_balancer as lb_types
|
from octavia.api.v1.types import load_balancer as lb_types
|
||||||
from octavia.common import constants
|
from octavia.common import constants
|
||||||
@ -178,9 +179,9 @@ class LoadBalancersController(base.BaseController):
|
|||||||
decides which controller, if any, should control be passed.
|
decides which controller, if any, should control be passed.
|
||||||
"""
|
"""
|
||||||
context = pecan.request.context.get('octavia_context')
|
context = pecan.request.context.get('octavia_context')
|
||||||
if lb_id and len(remainder) and (remainder[0] == 'listeners' or
|
|
||||||
remainder[0] == 'pools' or
|
possible_remainder = ('listeners', 'pools', 'delete_cascade', 'stats')
|
||||||
remainder[0] == 'delete_cascade'):
|
if lb_id and len(remainder) and (remainder[0] in possible_remainder):
|
||||||
controller = remainder[0]
|
controller = remainder[0]
|
||||||
remainder = remainder[1:]
|
remainder = remainder[1:]
|
||||||
db_lb = self.repositories.load_balancer.get(context.session,
|
db_lb = self.repositories.load_balancer.get(context.session,
|
||||||
@ -197,6 +198,9 @@ class LoadBalancersController(base.BaseController):
|
|||||||
load_balancer_id=db_lb.id), remainder
|
load_balancer_id=db_lb.id), remainder
|
||||||
elif (controller == 'delete_cascade'):
|
elif (controller == 'delete_cascade'):
|
||||||
return LBCascadeDeleteController(db_lb.id), ''
|
return LBCascadeDeleteController(db_lb.id), ''
|
||||||
|
elif (controller == 'stats'):
|
||||||
|
return lb_stats.LoadBalancerStatisticsController(
|
||||||
|
loadbalancer_id=db_lb.id), remainder
|
||||||
|
|
||||||
|
|
||||||
class LBCascadeDeleteController(LoadBalancersController):
|
class LBCascadeDeleteController(LoadBalancersController):
|
||||||
|
40
octavia/api/v1/controllers/load_balancer_statistics.py
Normal file
40
octavia/api/v1/controllers/load_balancer_statistics.py
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
# 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 pecan
|
||||||
|
from wsme import types as wtypes
|
||||||
|
from wsmeext import pecan as wsme_pecan
|
||||||
|
|
||||||
|
from octavia.api.v1.controllers import base
|
||||||
|
from octavia.api.v1.types import load_balancer_statistics as lb_types
|
||||||
|
from octavia.common import constants
|
||||||
|
from octavia.common import stats
|
||||||
|
|
||||||
|
|
||||||
|
class LoadBalancerStatisticsController(base.BaseController,
|
||||||
|
stats.StatsMixin):
|
||||||
|
|
||||||
|
def __init__(self, loadbalancer_id):
|
||||||
|
super(LoadBalancerStatisticsController, self).__init__()
|
||||||
|
self.loadbalancer_id = loadbalancer_id
|
||||||
|
|
||||||
|
@wsme_pecan.wsexpose(
|
||||||
|
{wtypes.text: lb_types.LoadBalancerStatisticsResponse})
|
||||||
|
def get(self):
|
||||||
|
"""Gets a single loadbalancer's statistics details."""
|
||||||
|
context = pecan.request.context.get('octavia_context')
|
||||||
|
data_stats = self.get_loadbalancer_stats(
|
||||||
|
context.session, self.loadbalancer_id)
|
||||||
|
return {constants.LOADBALANCER: self._convert_db_to_type(
|
||||||
|
data_stats, lb_types.LoadBalancerStatisticsResponse)}
|
50
octavia/api/v1/types/load_balancer_statistics.py
Normal file
50
octavia/api/v1/types/load_balancer_statistics.py
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
# Copyright 2016 Blue Box, an IBM Company
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
|
||||||
|
from wsme import types as wtypes
|
||||||
|
|
||||||
|
from octavia.api.v1.types import base
|
||||||
|
from octavia.api.v1.types import listener_statistics
|
||||||
|
|
||||||
|
|
||||||
|
class ListenerStatistics(listener_statistics.ListenerStatisticsResponse):
|
||||||
|
id = wtypes.wsattr(wtypes.UuidType())
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_data_model(cls, data_model, children=False):
|
||||||
|
ls_stats = super(ListenerStatistics, cls).from_data_model(
|
||||||
|
data_model, children=children)
|
||||||
|
ls_stats.id = data_model.listener_id
|
||||||
|
return ls_stats
|
||||||
|
|
||||||
|
|
||||||
|
class LoadBalancerStatisticsResponse(base.BaseType):
|
||||||
|
bytes_in = wtypes.wsattr(wtypes.IntegerType())
|
||||||
|
bytes_out = wtypes.wsattr(wtypes.IntegerType())
|
||||||
|
active_connections = wtypes.wsattr(wtypes.IntegerType())
|
||||||
|
total_connections = wtypes.wsattr(wtypes.IntegerType())
|
||||||
|
request_errors = wtypes.wsattr(wtypes.IntegerType())
|
||||||
|
listeners = wtypes.wsattr([ListenerStatistics])
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def from_data_model(cls, data_model, children=False):
|
||||||
|
lb_stats = super(LoadBalancerStatisticsResponse, cls).from_data_model(
|
||||||
|
data_model, children=children)
|
||||||
|
|
||||||
|
lb_stats.listeners = [
|
||||||
|
ListenerStatistics.from_data_model(
|
||||||
|
listener_dm, children=children)
|
||||||
|
for listener_dm in data_model.listeners
|
||||||
|
]
|
||||||
|
return lb_stats
|
@ -167,6 +167,28 @@ class ListenerStatistics(BaseDataModel):
|
|||||||
return self
|
return self
|
||||||
|
|
||||||
|
|
||||||
|
class LoadBalancerStatistics(BaseDataModel):
|
||||||
|
|
||||||
|
def __init__(self, bytes_in=0, bytes_out=0, active_connections=0,
|
||||||
|
total_connections=0, request_errors=0, listeners=None):
|
||||||
|
self.bytes_in = bytes_in
|
||||||
|
self.bytes_out = bytes_out
|
||||||
|
self.active_connections = active_connections
|
||||||
|
self.total_connections = total_connections
|
||||||
|
self.request_errors = request_errors
|
||||||
|
self.listeners = listeners or []
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
|
|
||||||
class HealthMonitor(BaseDataModel):
|
class HealthMonitor(BaseDataModel):
|
||||||
|
|
||||||
def __init__(self, id=None, project_id=None, pool_id=None, type=None,
|
def __init__(self, id=None, project_id=None, pool_id=None, type=None,
|
||||||
|
@ -29,6 +29,7 @@ class StatsMixin(object):
|
|||||||
super(StatsMixin, self).__init__()
|
super(StatsMixin, self).__init__()
|
||||||
self.listener_stats_repo = repo.ListenerStatisticsRepository()
|
self.listener_stats_repo = repo.ListenerStatisticsRepository()
|
||||||
self.repo_amphora = repo.AmphoraRepository()
|
self.repo_amphora = repo.AmphoraRepository()
|
||||||
|
self.repo_loadbalancer = repo.LoadBalancerRepository()
|
||||||
|
|
||||||
def get_listener_stats(self, session, listener_id):
|
def get_listener_stats(self, session, listener_id):
|
||||||
"""Gets the listener statistics data_models object."""
|
"""Gets the listener statistics data_models object."""
|
||||||
@ -48,3 +49,17 @@ class StatsMixin(object):
|
|||||||
if amp and amp.status == constants.AMPHORA_ALLOCATED:
|
if amp and amp.status == constants.AMPHORA_ALLOCATED:
|
||||||
statistics.active_connections += db_l.active_connections
|
statistics.active_connections += db_l.active_connections
|
||||||
return statistics
|
return statistics
|
||||||
|
|
||||||
|
def get_loadbalancer_stats(self, session, loadbalancer_id):
|
||||||
|
statistics = data_models.LoadBalancerStatistics()
|
||||||
|
lb_db = self.repo_loadbalancer.get(session, id=loadbalancer_id)
|
||||||
|
|
||||||
|
for listener in lb_db.listeners:
|
||||||
|
data = self.get_listener_stats(session, listener.id)
|
||||||
|
statistics.bytes_in += data.bytes_in
|
||||||
|
statistics.bytes_out += data.bytes_out
|
||||||
|
statistics.request_errors += data.request_errors
|
||||||
|
statistics.active_connections += data.active_connections
|
||||||
|
statistics.total_connections += data.total_connections
|
||||||
|
statistics.listeners.append(data)
|
||||||
|
return statistics
|
||||||
|
@ -217,6 +217,7 @@ class UpdateStatsDb(stats.StatsMixin):
|
|||||||
def __init__(self):
|
def __init__(self):
|
||||||
super(UpdateStatsDb, self).__init__()
|
super(UpdateStatsDb, self).__init__()
|
||||||
self.event_streamer = event_queue.EventStreamerNeutron()
|
self.event_streamer = event_queue.EventStreamerNeutron()
|
||||||
|
self.repo_listener = repo.ListenerRepository()
|
||||||
|
|
||||||
def emit(self, info_type, info_id, info_obj):
|
def emit(self, info_type, info_id, info_obj):
|
||||||
cnt = update_serializer.InfoContainer(info_type, info_id, info_obj)
|
cnt = update_serializer.InfoContainer(info_type, info_id, info_obj)
|
||||||
@ -272,3 +273,9 @@ class UpdateStatsDb(stats.StatsMixin):
|
|||||||
listener_stats = self.get_listener_stats(session, listener_id)
|
listener_stats = self.get_listener_stats(session, listener_id)
|
||||||
self.emit(
|
self.emit(
|
||||||
'listener_stats', listener_id, listener_stats.get_stats())
|
'listener_stats', listener_id, listener_stats.get_stats())
|
||||||
|
|
||||||
|
listener_db = self.repo_listener.get(session, id=listener_id)
|
||||||
|
lb_stats = self.get_loadbalancer_stats(
|
||||||
|
session, listener_db.load_balancer_id)
|
||||||
|
self.emit('loadbalancer_stats',
|
||||||
|
listener_db.load_balancer_id, lb_stats.get_stats())
|
||||||
|
@ -31,6 +31,7 @@ class BaseAPITest(base_db_test.OctaviaDBTestBase):
|
|||||||
BASE_PATH = '/v1'
|
BASE_PATH = '/v1'
|
||||||
LBS_PATH = '/loadbalancers'
|
LBS_PATH = '/loadbalancers'
|
||||||
LB_PATH = LBS_PATH + '/{lb_id}'
|
LB_PATH = LBS_PATH + '/{lb_id}'
|
||||||
|
LB_STATS_PATH = LB_PATH + '/stats'
|
||||||
LISTENERS_PATH = LB_PATH + '/listeners'
|
LISTENERS_PATH = LB_PATH + '/listeners'
|
||||||
LISTENER_PATH = LISTENERS_PATH + '/{listener_id}'
|
LISTENER_PATH = LISTENERS_PATH + '/{listener_id}'
|
||||||
LISTENER_STATS_PATH = LISTENER_PATH + '/stats'
|
LISTENER_STATS_PATH = LISTENER_PATH + '/stats'
|
||||||
|
@ -0,0 +1,56 @@
|
|||||||
|
# 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.
|
||||||
|
|
||||||
|
from octavia.common import constants
|
||||||
|
from octavia.tests.functional.api.v1 import base
|
||||||
|
|
||||||
|
from oslo_utils import uuidutils
|
||||||
|
|
||||||
|
|
||||||
|
class TestLoadBlancerStatistics(base.BaseAPITest):
|
||||||
|
FAKE_UUID_1 = uuidutils.generate_uuid()
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestLoadBlancerStatistics, self).setUp()
|
||||||
|
self.lb = self.create_load_balancer({})
|
||||||
|
self.set_lb_status(self.lb.get('id'))
|
||||||
|
self.listener = self.create_listener(self.lb.get('id'),
|
||||||
|
constants.PROTOCOL_HTTP, 80)
|
||||||
|
self.set_lb_status(self.lb.get('id'))
|
||||||
|
self.lb_path = self.LB_STATS_PATH.format(lb_id=self.lb.get('id'))
|
||||||
|
self.amphora = self.create_amphora(uuidutils.generate_uuid(),
|
||||||
|
self.lb.get('id'))
|
||||||
|
|
||||||
|
def test_get(self):
|
||||||
|
ls = self.create_listener_stats(listener_id=self.listener.get('id'),
|
||||||
|
amphora_id=self.amphora.id)
|
||||||
|
expected = {
|
||||||
|
'loadbalancer': {
|
||||||
|
'bytes_in': ls['bytes_in'],
|
||||||
|
'bytes_out': ls['bytes_out'],
|
||||||
|
'active_connections': ls['active_connections'],
|
||||||
|
'total_connections': ls['total_connections'],
|
||||||
|
'request_errors': ls['request_errors'],
|
||||||
|
'listeners': [
|
||||||
|
{'id': self.listener.get('id'),
|
||||||
|
'bytes_in': ls['bytes_in'],
|
||||||
|
'bytes_out': ls['bytes_out'],
|
||||||
|
'active_connections': ls['active_connections'],
|
||||||
|
'total_connections': ls['total_connections'],
|
||||||
|
'request_errors': ls['request_errors']}]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
response = self.get(self.lb_path)
|
||||||
|
response_body = response.json
|
||||||
|
self.assertEqual(expected, response_body)
|
@ -24,6 +24,7 @@ import sqlalchemy
|
|||||||
from octavia.common import constants
|
from octavia.common import constants
|
||||||
from octavia.common import data_models
|
from octavia.common import data_models
|
||||||
from octavia.controller.healthmanager import update_db
|
from octavia.controller.healthmanager import update_db
|
||||||
|
from octavia.db import models as db_models
|
||||||
from octavia.tests.unit import base
|
from octavia.tests.unit import base
|
||||||
|
|
||||||
|
|
||||||
@ -468,14 +469,33 @@ class TestUpdateStatsDb(base.TestCase):
|
|||||||
total_connections=random.randrange(1000000000),
|
total_connections=random.randrange(1000000000),
|
||||||
request_errors=random.randrange(1000000000))
|
request_errors=random.randrange(1000000000))
|
||||||
|
|
||||||
self.sm.get_listener_stats = mock.MagicMock(
|
self.sm.get_listener_stats = mock.MagicMock()
|
||||||
return_value=self.listener_stats)
|
self.sm.get_listener_stats.return_value = self.listener_stats
|
||||||
|
|
||||||
|
self.loadbalancer_id = uuidutils.generate_uuid()
|
||||||
|
self.amphora_id = uuidutils.generate_uuid()
|
||||||
|
self.listener_id = uuidutils.generate_uuid()
|
||||||
|
|
||||||
|
self.listener = db_models.Listener(
|
||||||
|
load_balancer_id=self.loadbalancer_id)
|
||||||
|
|
||||||
|
self.listener_repo = mock.MagicMock()
|
||||||
|
self.sm.repo_listener = self.listener_repo
|
||||||
|
self.sm.repo_listener.get.return_value = self.listener
|
||||||
|
|
||||||
|
self.loadbalancer_repo = mock.MagicMock()
|
||||||
|
self.sm.repo_loadbalancer = self.loadbalancer_repo
|
||||||
|
|
||||||
|
self.loadbalancer = db_models.LoadBalancer(
|
||||||
|
id=self.loadbalancer_id,
|
||||||
|
listeners=[self.listener])
|
||||||
|
self.loadbalancer_repo.get.return_value = self.loadbalancer
|
||||||
|
|
||||||
@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):
|
||||||
|
|
||||||
health = {
|
health = {
|
||||||
"id": self.loadbalancer_id,
|
"id": self.amphora_id,
|
||||||
"listeners": {
|
"listeners": {
|
||||||
self.listener_id: {
|
self.listener_id: {
|
||||||
"status": constants.OPEN,
|
"status": constants.OPEN,
|
||||||
@ -501,13 +521,13 @@ class TestUpdateStatsDb(base.TestCase):
|
|||||||
self.sm.update_stats(health)
|
self.sm.update_stats(health)
|
||||||
|
|
||||||
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.amphora_id,
|
||||||
bytes_in=self.listener_stats.bytes_in,
|
bytes_in=self.listener_stats.bytes_in,
|
||||||
bytes_out=self.listener_stats.bytes_out,
|
bytes_out=self.listener_stats.bytes_out,
|
||||||
active_connections=self.listener_stats.active_connections,
|
active_connections=self.listener_stats.active_connections,
|
||||||
total_connections=self.listener_stats.total_connections,
|
total_connections=self.listener_stats.total_connections,
|
||||||
request_errors=self.listener_stats.request_errors)
|
request_errors=self.listener_stats.request_errors)
|
||||||
self.event_client.cast.assert_called_once_with(
|
self.event_client.cast.assert_any_call(
|
||||||
{}, 'update_info', container={
|
{}, 'update_info', container={
|
||||||
'info_type': 'listener_stats',
|
'info_type': 'listener_stats',
|
||||||
'info_id': self.listener_id,
|
'info_id': self.listener_id,
|
||||||
@ -519,3 +539,17 @@ class TestUpdateStatsDb(base.TestCase):
|
|||||||
self.listener_stats.active_connections,
|
self.listener_stats.active_connections,
|
||||||
'bytes_out': self.listener_stats.bytes_out,
|
'bytes_out': self.listener_stats.bytes_out,
|
||||||
'request_errors': self.listener_stats.request_errors}})
|
'request_errors': self.listener_stats.request_errors}})
|
||||||
|
|
||||||
|
self.event_client.cast.assert_any_call(
|
||||||
|
{}, 'update_info',
|
||||||
|
container={
|
||||||
|
'info_type': 'loadbalancer_stats',
|
||||||
|
'info_id': self.loadbalancer_id,
|
||||||
|
'info_payload': {
|
||||||
|
'bytes_in': self.listener_stats.bytes_in,
|
||||||
|
'total_connections':
|
||||||
|
self.listener_stats.total_connections,
|
||||||
|
'active_connections':
|
||||||
|
self.listener_stats.active_connections,
|
||||||
|
'bytes_out': self.listener_stats.bytes_out,
|
||||||
|
'request_errors': self.listener_stats.request_errors}})
|
||||||
|
Loading…
Reference in New Issue
Block a user