speed up GET scheduler-stats/pools/detail

return a cached host state map, which gets updated
on each scheduling
Change-Id: Ia258164a43fc15f83dfadfc1133ab4faa972e41c
Closes-Bug: #1804659
This commit is contained in:
Maurice Schreiber 2018-09-04 15:24:06 +02:00 committed by Tom Barron
parent 1a658eb187
commit 32a19fd56c
9 changed files with 34 additions and 19 deletions

View File

@ -71,7 +71,8 @@ class SchedulerStatsController(wsgi.Controller):
msg = _("Share type %s not found.") % req_share_type msg = _("Share type %s not found.") % req_share_type
raise exc.HTTPBadRequest(explanation=msg) raise exc.HTTPBadRequest(explanation=msg)
pools = self.scheduler_api.get_pools(context, filters=search_opts) pools = self.scheduler_api.get_pools(context, filters=search_opts,
cached=True)
detail = (action == 'detail') detail = (action == 'detail')
return self._view_builder.pools(pools, detail=detail) return self._view_builder.pools(pools, detail=detail)

View File

@ -45,8 +45,8 @@ class FilterScheduler(base.Scheduler):
"""Fetch options dictionary. Broken out for testing.""" """Fetch options dictionary. Broken out for testing."""
return self.options.get_configuration() return self.options.get_configuration()
def get_pools(self, context, filters): def get_pools(self, context, filters, cached):
return self.host_manager.get_pools(context, filters) return self.host_manager.get_pools(context, filters, cached)
def _post_select_populate_filter_properties(self, filter_properties, def _post_select_populate_filter_properties(self, filter_properties,
host_state): host_state):

View File

@ -626,9 +626,10 @@ class HostManager(object):
return all_pools.values() return all_pools.values()
def get_pools(self, context, filters=None): def get_pools(self, context, filters=None, cached=False):
"""Returns a dict of all pools on all hosts HostManager knows about.""" """Returns a dict of all pools on all hosts HostManager knows about."""
self._update_host_state_map(context) if not cached or not self.host_state_map:
self._update_host_state_map(context)
all_pools = [] all_pools = []
for host, host_state in self.host_state_map.items(): for host, host_state in self.host_state_map.items():

View File

@ -62,7 +62,7 @@ MAPPING = {
class SchedulerManager(manager.Manager): class SchedulerManager(manager.Manager):
"""Chooses a host to create shares.""" """Chooses a host to create shares."""
RPC_API_VERSION = '1.8' RPC_API_VERSION = '1.9'
def __init__(self, scheduler_driver=None, service_name=None, def __init__(self, scheduler_driver=None, service_name=None,
*args, **kwargs): *args, **kwargs):
@ -120,9 +120,9 @@ class SchedulerManager(manager.Manager):
'create_share', {'status': constants.STATUS_ERROR}, 'create_share', {'status': constants.STATUS_ERROR},
context, ex, request_spec) context, ex, request_spec)
def get_pools(self, context, filters=None): def get_pools(self, context, filters=None, cached=False):
"""Get active pools from the scheduler's cache.""" """Get active pools from the scheduler's cache."""
return self.driver.get_pools(context, filters) return self.driver.get_pools(context, filters, cached)
def manage_share(self, context, share_id, driver_options, request_spec, def manage_share(self, context, share_id, driver_options, request_spec,
filter_properties=None): filter_properties=None):

View File

@ -40,9 +40,10 @@ class SchedulerAPI(object):
1.6 - Add manage_share 1.6 - Add manage_share
1.7 - Updated migrate_share_to_host method with new parameters 1.7 - Updated migrate_share_to_host method with new parameters
1.8 - Rename create_consistency_group -> create_share_group method 1.8 - Rename create_consistency_group -> create_share_group method
1.9 - Add cached parameter to get_pools method
""" """
RPC_API_VERSION = '1.8' RPC_API_VERSION = '1.9'
def __init__(self): def __init__(self):
super(SchedulerAPI, self).__init__() super(SchedulerAPI, self).__init__()
@ -69,9 +70,10 @@ class SchedulerAPI(object):
host=host, host=host,
capabilities=capabilities) capabilities=capabilities)
def get_pools(self, context, filters=None): def get_pools(self, context, filters=None, cached=False):
call_context = self.client.prepare(version='1.1') call_context = self.client.prepare(version='1.9')
return call_context.call(context, 'get_pools', filters=filters) return call_context.call(context, 'get_pools', filters=filters,
cached=cached)
def create_share_group(self, context, share_group_id, request_spec=None, def create_share_group(self, context, share_group_id, request_spec=None,
filter_properties=None): filter_properties=None):

View File

@ -101,7 +101,8 @@ class SchedulerStatsControllerTestCase(test.TestCase):
} }
self.assertDictMatch(result, expected) self.assertDictMatch(result, expected)
mock_get_pools.assert_called_once_with(self.ctxt, filters={}) mock_get_pools.assert_called_once_with(self.ctxt, filters={},
cached=True)
self.mock_policy_check.assert_called_once_with( self.mock_policy_check.assert_called_once_with(
self.ctxt, self.resource_name, 'index') self.ctxt, self.resource_name, 'index')
@ -148,7 +149,8 @@ class SchedulerStatsControllerTestCase(test.TestCase):
self.assertDictMatch(result, expected_result) self.assertDictMatch(result, expected_result)
mock_get_pools.assert_called_once_with(self.ctxt, mock_get_pools.assert_called_once_with(self.ctxt,
filters=expected_filters) filters=expected_filters,
cached=True)
@ddt.data(('index', False, True), @ddt.data(('index', False, True),
('index', False, False), ('index', False, False),
@ -210,7 +212,8 @@ class SchedulerStatsControllerTestCase(test.TestCase):
self.assertDictMatch(result, expected_result) self.assertDictMatch(result, expected_result)
mock_get_pools.assert_called_once_with(self.ctxt, mock_get_pools.assert_called_once_with(self.ctxt,
filters=expected_filters) filters=expected_filters,
cached=True)
@ddt.data('index', 'detail') @ddt.data('index', 'detail')
def test_pools_with_share_type_not_found(self, action): def test_pools_with_share_type_not_found(self, action):
@ -271,7 +274,8 @@ class SchedulerStatsControllerTestCase(test.TestCase):
self.assertDictMatch(result, expected) self.assertDictMatch(result, expected)
mock_get_pools.assert_called_once_with(self.ctxt, mock_get_pools.assert_called_once_with(self.ctxt,
filters=expected_filters) filters=expected_filters,
cached=True)
self.mock_policy_check.assert_called_once_with( self.mock_policy_check.assert_called_once_with(
self.ctxt, self.resource_name, 'index') self.ctxt, self.resource_name, 'index')
@ -323,7 +327,8 @@ class SchedulerStatsControllerTestCase(test.TestCase):
} }
self.assertDictMatch(expected, result) self.assertDictMatch(expected, result)
mock_get_pools.assert_called_once_with(self.ctxt, filters={}) mock_get_pools.assert_called_once_with(self.ctxt, filters={},
cached=True)
self.mock_policy_check.assert_called_once_with( self.mock_policy_check.assert_called_once_with(
self.ctxt, self.resource_name, 'detail') self.ctxt, self.resource_name, 'detail')

View File

@ -224,7 +224,8 @@ class SchedulerManagerTestCase(test.TestCase):
result = self.manager.get_pools(self.context, filters='fake_filters') result = self.manager.get_pools(self.context, filters='fake_filters')
mock_get_pools.assert_called_once_with(self.context, 'fake_filters') mock_get_pools.assert_called_once_with(self.context, 'fake_filters',
False)
self.assertEqual('fake_pools', result) self.assertEqual('fake_pools', result)
@mock.patch.object(db, 'share_group_update', mock.Mock()) @mock.patch.object(db, 'share_group_update', mock.Mock())

View File

@ -88,7 +88,7 @@ class SchedulerRpcAPITestCase(test.TestCase):
self._test_scheduler_api('get_pools', self._test_scheduler_api('get_pools',
rpc_method='call', rpc_method='call',
filters=None, filters=None,
version='1.1') version='1.9')
def test_create_share_group(self): def test_create_share_group(self):
self._test_scheduler_api('create_share_group', self._test_scheduler_api('create_share_group',

View File

@ -0,0 +1,5 @@
---
fixes:
- |
Added caching of host state map to speed up calls for
scheduler-stats/pools/detail.