Make a subunit2sql.target extension

Just having this installed should make subunit2sql shove any
counters.json it finds through to statsd.
This commit is contained in:
Clint Byrum 2015-11-05 23:56:26 -08:00
parent e7d63bfd10
commit f920b45956
5 changed files with 77 additions and 27 deletions

View File

@ -10,16 +10,24 @@
# See the License for the specific language governing permissions and
# limitations under the License.
import tempfile
import json
from oslo_config import cfg
import statsd
try:
import statsd
except ImportError:
statsd = None
import testtools
OPTS_GROUP = cfg.OptGroup(name='counters2statsd', title='Counters2Statsd')
OPTS = [
cfg.StrOpt('host', help='Statsd host to connect to', default=None),
cfg.IntOpt('port', help='Port on statsd host to connect to', default=None),
cfg.StrOpt('host', help='Statsd host to connect to', default='localhost'),
cfg.IntOpt('port', help='Port on statsd host to connect to', default=8125),
cfg.StrOpt('prefix', help='Prefix to add to stats', default=None),
cfg.BoolOpt('enabled', help='Set to false to disable this plugin',
default=True)
]
_statsd_client = None
@ -28,32 +36,70 @@ _statsd_client = None
def get_statsd_client():
global _statsd_client
if _statsd_client is None:
cfg.CONF.register_group(OPTS_GROUP)
cfg.CONF.register_opts(OPTS, group=OPTS_GROUP)
_statsd_client = statsd.StatsClient(cfg.CONF.counters2statsd.host,
cfg.CONF.counters2statsd.port,
cfg.CONF.counters2statsd.prefix)
return _statsd_client
def add_test_run_attachments(attachments, test_run_id, session):
for attachment in attachments:
try:
counters = json.loads(attachment)
except ValueError:
continue
if not isinstance(counters, dict):
continue
if '__counters_meta__' not in counters:
continue
class AttachmentResult(testtools.StreamResult):
"""Keeps track of top level results with StreamToDict drops.
We use a SpooledTemporaryFile to keep it performant with smaller files
but to ensure we don't use up tons of RAM. Anything over 1MB will be
spooled out to disk.
"""
@classmethod
def enabled(cls):
cfg.CONF.register_group(OPTS_GROUP)
cfg.CONF.register_opts(OPTS, group=OPTS_GROUP)
cfg.CONF.register_cli_opts(OPTS, group=OPTS_GROUP)
return bool(statsd)
def __init__(self):
super(AttachmentResult, self).__init__()
self.attachments = {}
def status(self, test_id=None, test_status=None, test_tags=None,
runnable=True, file_name=None, file_bytes=None, eof=False,
mime_type=None, route_code=None, timestamp=None):
if not cfg.CONF.counters2statsd.enabled:
return
if test_id is not None:
return
if not file_name:
return
if file_name not in self.attachments:
self.attachments[file_name] = tempfile.SpooledTemporaryFile(
max_size=2 ** 30)
self.attachments[file_name].write(file_bytes)
if eof:
self.attachments[file_name].seek(0)
def stopTestRun(self):
if not cfg.CONF.counters2statsd.enabled:
return
client = get_statsd_client()
for groupname, values in counters.items():
if not isinstance(values, dict):
for file_name, attachment in self.attachments.items():
if file_name != 'counters.json':
continue
for k, v in values.items():
k = '{}.{}'.format(groupname, k)
try:
try:
v = int(v)
except ValueError:
attachment.seek(0)
counters = json.loads(attachment.read().decode('utf-8'))
except AttributeError:
counters = json.loads(attachment)
except ValueError:
continue
if not isinstance(counters, dict):
continue
for groupname, values in counters.items():
if not isinstance(values, dict):
continue
client.incr(k, v)
for k, v in values.items():
k = '{}.{}'.format(groupname, k)
try:
v = int(v)
except ValueError:
continue
client.incr(k, v)

View File

@ -34,8 +34,10 @@ class TestOpenStackQaTols(base.TestCase):
mock_client.incr = mock.MagicMock('statds_incr')
statsd_mock.return_value = mock_client
fake_counters = {'mysql': {'Queries': 10}}
fake_counters['__counters_meta__'] = {}
fake_counters = json.dumps(fake_counters)
counters2statsd.add_test_run_attachments([fake_counters], 'foo', None)
statsd_mock.assert_called_with(None, None, None)
fake_counters = json.dumps(fake_counters).encode('utf-8')
self.assertTrue(counters2statsd.AttachmentResult.enabled())
result = counters2statsd.AttachmentResult()
result.status(file_name='counters.json', file_bytes=fake_counters)
result.stopTestRun()
statsd_mock.assert_called_with('localhost', 8125, None)
mock_client.incr.assert_called_with('mysql.Queries', 10)

View File

@ -6,3 +6,4 @@ pbr>=1.6
PyMySQL>=0.6.2 # MIT License
statsd>=1.0.0,<3.0
oslo.config>=1.4.0.0a3
testtools>=1.4.0

View File

@ -22,6 +22,8 @@ classifier =
[entry_points]
console_scripts =
os-qa-counters = openstack_qa_tools.collect:main
subunit2sql.target =
openstack_qa_statsd = openstack_qa_tools.counters2statsd:AttachmentResult
[files]
packages =

View File

@ -12,5 +12,4 @@ oslosphinx>=2.5.0 # Apache-2.0
oslotest>=1.10.0 # Apache-2.0
testrepository>=0.0.18
testscenarios>=0.4
testtools>=1.4.0
mock>=1.2