Add new meters for swift
This patch adds new meters: -storage.containers.objects -storage.containers.objects.size. These meters are implemented through a polling Swift blueprint: https://wiki.openstack.org/wiki/Ceilometer/blueprints/pollster-swift Change-Id: Ic1f3aaf351db0b99392eb13d4f31f2732f87f816
This commit is contained in:
parent
19e06732b7
commit
7b9cc2a5ff
@ -47,15 +47,19 @@ cfg.CONF.register_opts(OPTS)
|
|||||||
class _Base(plugin.PollsterBase):
|
class _Base(plugin.PollsterBase):
|
||||||
|
|
||||||
CACHE_KEY_TENANT = 'tenants'
|
CACHE_KEY_TENANT = 'tenants'
|
||||||
CACHE_KEY_HEAD = 'swift.head_account'
|
METHOD = 'head'
|
||||||
|
|
||||||
|
@property
|
||||||
|
def CACHE_KEY_METHOD(self):
|
||||||
|
return 'swift.%s_account' % self.METHOD
|
||||||
|
|
||||||
def _iter_accounts(self, ksclient, cache):
|
def _iter_accounts(self, ksclient, cache):
|
||||||
if self.CACHE_KEY_TENANT not in cache:
|
if self.CACHE_KEY_TENANT not in cache:
|
||||||
cache[self.CACHE_KEY_TENANT] = ksclient.tenants.list()
|
cache[self.CACHE_KEY_TENANT] = ksclient.tenants.list()
|
||||||
if self.CACHE_KEY_HEAD not in cache:
|
if self.CACHE_KEY_METHOD not in cache:
|
||||||
cache[self.CACHE_KEY_HEAD] = list(self._get_account_info(ksclient,
|
cache[self.CACHE_KEY_METHOD] = list(self._get_account_info(
|
||||||
cache))
|
ksclient, cache))
|
||||||
return iter(cache[self.CACHE_KEY_HEAD])
|
return iter(cache[self.CACHE_KEY_METHOD])
|
||||||
|
|
||||||
def _get_account_info(self, ksclient, cache):
|
def _get_account_info(self, ksclient, cache):
|
||||||
try:
|
try:
|
||||||
@ -67,7 +71,9 @@ class _Base(plugin.PollsterBase):
|
|||||||
raise StopIteration()
|
raise StopIteration()
|
||||||
|
|
||||||
for t in cache[self.CACHE_KEY_TENANT]:
|
for t in cache[self.CACHE_KEY_TENANT]:
|
||||||
yield (t.id, swift.head_account(self._neaten_url(endpoint, t.id),
|
api_method = '%s_account' % self.METHOD
|
||||||
|
yield (t.id, getattr(swift, api_method)
|
||||||
|
(self._neaten_url(endpoint, t.id),
|
||||||
ksclient.auth_token))
|
ksclient.auth_token))
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
@ -133,3 +139,49 @@ class ObjectsContainersPollster(_Base):
|
|||||||
timestamp=timeutils.isotime(),
|
timestamp=timeutils.isotime(),
|
||||||
resource_metadata=None,
|
resource_metadata=None,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class ContainersObjectsPollster(_Base):
|
||||||
|
"""Get info about containers using Swift API
|
||||||
|
"""
|
||||||
|
|
||||||
|
METHOD = 'get'
|
||||||
|
|
||||||
|
def get_samples(self, manager, cache):
|
||||||
|
for project, account in self._iter_accounts(manager.keystone, cache):
|
||||||
|
containers_info = account[1]
|
||||||
|
for container in containers_info:
|
||||||
|
yield sample.Sample(
|
||||||
|
name='storage.containers.objects',
|
||||||
|
type=sample.TYPE_GAUGE,
|
||||||
|
volume=int(container['count']),
|
||||||
|
unit='object',
|
||||||
|
user_id=None,
|
||||||
|
project_id=project,
|
||||||
|
resource_id=project + '/' + container['name'],
|
||||||
|
timestamp=timeutils.isotime(),
|
||||||
|
resource_metadata=None,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class ContainersSizePollster(_Base):
|
||||||
|
"""Get info about containers using Swift API
|
||||||
|
"""
|
||||||
|
|
||||||
|
METHOD = 'get'
|
||||||
|
|
||||||
|
def get_samples(self, manager, cache):
|
||||||
|
for project, account in self._iter_accounts(manager.keystone, cache):
|
||||||
|
containers_info = account[1]
|
||||||
|
for container in containers_info:
|
||||||
|
yield sample.Sample(
|
||||||
|
name='storage.containers.objects.size',
|
||||||
|
type=sample.TYPE_GAUGE,
|
||||||
|
volume=int(container['bytes']),
|
||||||
|
unit='B',
|
||||||
|
user_id=None,
|
||||||
|
project_id=project,
|
||||||
|
resource_id=project + '/' + container['name'],
|
||||||
|
timestamp=timeutils.isotime(),
|
||||||
|
resource_metadata=None,
|
||||||
|
)
|
||||||
|
@ -31,7 +31,7 @@ from ceilometer.openstack.common import test
|
|||||||
|
|
||||||
load_tests = testscenarios.load_tests_apply_scenarios
|
load_tests = testscenarios.load_tests_apply_scenarios
|
||||||
|
|
||||||
ACCOUNTS = [('tenant-000', {'x-account-object-count': 12,
|
HEAD_ACCOUNTS = [('tenant-000', {'x-account-object-count': 12,
|
||||||
'x-account-bytes-used': 321321321,
|
'x-account-bytes-used': 321321321,
|
||||||
'x-account-container-count': 7,
|
'x-account-container-count': 7,
|
||||||
}),
|
}),
|
||||||
@ -40,6 +40,22 @@ ACCOUNTS = [('tenant-000', {'x-account-object-count': 12,
|
|||||||
'x-account-container-count': 17,
|
'x-account-container-count': 17,
|
||||||
})]
|
})]
|
||||||
|
|
||||||
|
GET_ACCOUNTS = [('tenant-002', ({'x-account-object-count': 10,
|
||||||
|
'x-account-bytes-used': 123123,
|
||||||
|
'x-account-container-count': 2,
|
||||||
|
},
|
||||||
|
[{'count': 10,
|
||||||
|
'bytes': 123123,
|
||||||
|
'name': 'my_container'},
|
||||||
|
{'count': 0,
|
||||||
|
'bytes': 0,
|
||||||
|
'name': 'new_container'
|
||||||
|
}])),
|
||||||
|
('tenant-003', ({'x-account-object-count': 0,
|
||||||
|
'x-account-bytes-used': 0,
|
||||||
|
'x-account-container-count': 0,
|
||||||
|
}, [])), ]
|
||||||
|
|
||||||
|
|
||||||
class TestManager(manager.AgentManager):
|
class TestManager(manager.AgentManager):
|
||||||
|
|
||||||
@ -59,6 +75,10 @@ class TestSwiftPollster(test.BaseTestCase):
|
|||||||
{'factory': swift.ObjectsSizePollster}),
|
{'factory': swift.ObjectsSizePollster}),
|
||||||
('storage.objects.containers',
|
('storage.objects.containers',
|
||||||
{'factory': swift.ObjectsContainersPollster}),
|
{'factory': swift.ObjectsContainersPollster}),
|
||||||
|
('storage.containers.objects',
|
||||||
|
{'factory': swift.ContainersObjectsPollster}),
|
||||||
|
('storage.containers.objects.size',
|
||||||
|
{'factory': swift.ContainersSizePollster}),
|
||||||
]
|
]
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
@ -66,7 +86,7 @@ class TestSwiftPollster(test.BaseTestCase):
|
|||||||
raise exceptions.EndpointNotFound("Fake keystone exception")
|
raise exceptions.EndpointNotFound("Fake keystone exception")
|
||||||
|
|
||||||
def fake_iter_accounts(self, ksclient, cache):
|
def fake_iter_accounts(self, ksclient, cache):
|
||||||
for i in ACCOUNTS:
|
for i in self.ACCOUNTS:
|
||||||
yield i
|
yield i
|
||||||
|
|
||||||
@mock.patch('ceilometer.pipeline.setup_pipeline', mock.MagicMock())
|
@mock.patch('ceilometer.pipeline.setup_pipeline', mock.MagicMock())
|
||||||
@ -75,6 +95,11 @@ class TestSwiftPollster(test.BaseTestCase):
|
|||||||
self.pollster = self.factory()
|
self.pollster = self.factory()
|
||||||
self.manager = TestManager()
|
self.manager = TestManager()
|
||||||
|
|
||||||
|
if self.pollster.CACHE_KEY_METHOD == 'swift.head_account':
|
||||||
|
self.ACCOUNTS = HEAD_ACCOUNTS
|
||||||
|
else:
|
||||||
|
self.ACCOUNTS = GET_ACCOUNTS
|
||||||
|
|
||||||
def test_iter_accounts_no_cache(self):
|
def test_iter_accounts_no_cache(self):
|
||||||
cache = {}
|
cache = {}
|
||||||
with PatchObject(self.factory, '_get_account_info',
|
with PatchObject(self.factory, '_get_account_info',
|
||||||
@ -82,7 +107,7 @@ class TestSwiftPollster(test.BaseTestCase):
|
|||||||
data = list(self.pollster._iter_accounts(mock.Mock(), cache))
|
data = list(self.pollster._iter_accounts(mock.Mock(), cache))
|
||||||
|
|
||||||
self.assertTrue(self.pollster.CACHE_KEY_TENANT in cache)
|
self.assertTrue(self.pollster.CACHE_KEY_TENANT in cache)
|
||||||
self.assertTrue(self.pollster.CACHE_KEY_HEAD in cache)
|
self.assertTrue(self.pollster.CACHE_KEY_METHOD in cache)
|
||||||
self.assertEqual(data, [])
|
self.assertEqual(data, [])
|
||||||
|
|
||||||
def test_iter_accounts_tenants_cached(self):
|
def test_iter_accounts_tenants_cached(self):
|
||||||
@ -94,16 +119,18 @@ class TestSwiftPollster(test.BaseTestCase):
|
|||||||
'should not be called',
|
'should not be called',
|
||||||
)
|
)
|
||||||
|
|
||||||
with PatchObject(swift_client, 'head_account', new=ksclient):
|
api_method = '%s_account' % self.pollster.METHOD
|
||||||
|
with PatchObject(swift_client, api_method, new=ksclient):
|
||||||
with PatchObject(self.factory, '_neaten_url'):
|
with PatchObject(self.factory, '_neaten_url'):
|
||||||
Tenant = collections.namedtuple('Tenant', 'id')
|
Tenant = collections.namedtuple('Tenant', 'id')
|
||||||
cache = {
|
cache = {
|
||||||
self.pollster.CACHE_KEY_TENANT: [Tenant(ACCOUNTS[0][0])],
|
self.pollster.CACHE_KEY_TENANT: [
|
||||||
|
Tenant(self.ACCOUNTS[0][0])
|
||||||
|
],
|
||||||
}
|
}
|
||||||
data = list(self.pollster._iter_accounts(mock.Mock(), cache))
|
data = list(self.pollster._iter_accounts(mock.Mock(), cache))
|
||||||
|
self.assertTrue(self.pollster.CACHE_KEY_METHOD in cache)
|
||||||
self.assertTrue(self.pollster.CACHE_KEY_HEAD in cache)
|
self.assertEqual(data[0][0], self.ACCOUNTS[0][0])
|
||||||
self.assertEqual(data[0][0], ACCOUNTS[0][0])
|
|
||||||
|
|
||||||
def test_neaten_url(self):
|
def test_neaten_url(self):
|
||||||
test_endpoint = 'http://127.0.0.1:8080'
|
test_endpoint = 'http://127.0.0.1:8080'
|
||||||
|
@ -144,16 +144,18 @@ Make sure Cinder is properly configured first: see :ref:`installing_manually`.
|
|||||||
Object Storage (Swift)
|
Object Storage (Swift)
|
||||||
======================
|
======================
|
||||||
|
|
||||||
============================== ========== ========== ======== ============ ==============================================
|
=============================== ========== ========== =========== ============ ==========================================
|
||||||
Name Type Volume Resource Origin Note
|
Name Type Volume Resource Origin Note
|
||||||
============================== ========== ========== ======== ============ ==============================================
|
=============================== ========== ========== =========== ============ ==========================================
|
||||||
storage.objects Gauge object store ID pollster Number of objects
|
storage.objects Gauge object store ID pollster Number of objects
|
||||||
storage.objects.size Gauge B store ID pollster Total size of stored objects
|
storage.objects.size Gauge B store ID pollster Total size of stored objects
|
||||||
storage.objects.containers Gauge container store ID pollster Number of containers
|
storage.objects.containers Gauge container store ID pollster Number of containers
|
||||||
storage.objects.incoming.bytes Delta B store ID notification Number of incoming bytes
|
storage.objects.incoming.bytes Delta B store ID notification Number of incoming bytes
|
||||||
storage.objects.outgoing.bytes Delta B store ID notification Number of outgoing bytes
|
storage.objects.outgoing.bytes Delta B store ID notification Number of outgoing bytes
|
||||||
storage.api.request Delta request store ID notification Number of API requests against swift
|
storage.api.request Delta request store ID notification Number of API requests against swift
|
||||||
============================== ========== ========== ======== ============ ==============================================
|
storage.containers.objects Gauge object str ID/cont pollster Number of objects in container
|
||||||
|
storage.containers.objects.size Gauge B str ID/cont pollster Total size of stored objects in container
|
||||||
|
=============================== ========== ========== =========== ============ ==========================================
|
||||||
|
|
||||||
In order to use storage.objects.incoming.bytes and storage.outgoing.bytes, one must configure
|
In order to use storage.objects.incoming.bytes and storage.outgoing.bytes, one must configure
|
||||||
Swift as described in :ref:`installing_manually`. Note that they may not be
|
Swift as described in :ref:`installing_manually`. Note that they may not be
|
||||||
|
@ -70,6 +70,8 @@ ceilometer.poll.central =
|
|||||||
ip.floating = ceilometer.network.floatingip:FloatingIPPollster
|
ip.floating = ceilometer.network.floatingip:FloatingIPPollster
|
||||||
image = ceilometer.image.glance:ImagePollster
|
image = ceilometer.image.glance:ImagePollster
|
||||||
image.size = ceilometer.image.glance:ImageSizePollster
|
image.size = ceilometer.image.glance:ImageSizePollster
|
||||||
|
storage.containers.objects = ceilometer.objectstore.swift:ContainersObjectsPollster
|
||||||
|
storage.containers.objects.size = ceilometer.objectstore.swift:ContainersSizePollster
|
||||||
storage.objects = ceilometer.objectstore.swift:ObjectsPollster
|
storage.objects = ceilometer.objectstore.swift:ObjectsPollster
|
||||||
storage.objects.size = ceilometer.objectstore.swift:ObjectsSizePollster
|
storage.objects.size = ceilometer.objectstore.swift:ObjectsSizePollster
|
||||||
storage.objects.containers = ceilometer.objectstore.swift:ObjectsContainersPollster
|
storage.objects.containers = ceilometer.objectstore.swift:ObjectsContainersPollster
|
||||||
|
Loading…
Reference in New Issue
Block a user