Merge pull request #35 from dave-shawley/add-close

Add StatsDCollector.close and some docs
This commit is contained in:
Andrew Rabert 2019-09-02 22:32:00 -04:00 committed by GitHub
commit 54fc1ff50b
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
8 changed files with 117 additions and 28 deletions

View file

@ -49,6 +49,18 @@ Statsd Implementation
.. autoclass:: sprockets.mixins.metrics.statsd.StatsdMixin
:members:
.. autoclass:: sprockets.mixins.metrics.statsd.StatsDCollector
:members:
Application Functions
---------------------
Before you can use the mixin, you have to install the client by calling
the ``install`` function on your application instance.
.. autofunction:: sprockets.mixins.metrics.statsd.install
.. autofunction:: sprockets.mixins.metrics.statsd.get_client
Testing Helpers
---------------
*So who actually tests that their metrics are emitted as they expect?*

View file

@ -7,6 +7,8 @@ Release History
---------------
- Add configuration documentation
- Exclude Tornado >6 (as-yet-unreleased version)
- Add :func:`sprockets.mixins.metrics.statsd.get_client` function
- Add :meth:`sprockets.mixins.metrics.statsd.StatsDCollector.close` method
`4.0.0`_ (06-Feb-2019)
----------------------

View file

@ -1,3 +1,3 @@
coverage==4.5.2
flake8==3.6.0
coverage==4.5.4
flake8==3.7.8
nose==1.3.7

View file

@ -8,6 +8,9 @@ warning-is-error = 1
[check]
strict = 1
[coverage:report]
show_missing = 1
[nosetests]
cover-package = sprockets.mixins.metrics
cover-branches = 1

View file

@ -32,6 +32,7 @@ setuptools.setup(
'Natural Language :: English',
'Operating System :: OS Independent',
'Programming Language :: Python :: 3.7',
'Programming Language :: Python :: 3.8',
'Programming Language :: Python :: Implementation :: CPython',
'Topic :: Internet :: WWW/HTTP',
'Topic :: Software Development :: Libraries',

View file

@ -29,7 +29,9 @@ class StatsdMixin:
:param path: elements of the metric path to record
"""
self.application.statsd.send(path, duration * 1000.0, 'ms')
client = get_client(self.application)
if client is not None:
client.send(path, duration * 1000.0, 'ms')
def increase_counter(self, *path, **kwargs):
"""Increase a counter.
@ -45,7 +47,9 @@ class StatsdMixin:
omitted, the counter is increased by one.
"""
self.application.statsd.send(path, kwargs.get('amount', '1'), 'c')
client = get_client(self.application)
if client is not None:
client.send(path, kwargs.get('amount', '1'), 'c')
@contextlib.contextmanager
def execution_timer(self, *path):
@ -86,10 +90,9 @@ class StatsdMixin:
class StatsDCollector:
"""Collects and submits stats to StatsD.
This class should be constructed using the
:meth:`~sprockets.mixins.statsd.install` method. When installed,
it is attached to the :class:`~tornado.web.Application` instance
for your web application.
This class should be constructed using the :func:`.install` function.
When installed, it is attached to the :class:`~tornado.web.Application`
instance for your web application.
:param str host: The StatsD host
:param str port: The StatsD port
@ -110,6 +113,7 @@ class StatsDCollector:
self._namespace = namespace
self._prepend_metric_type = prepend_metric_type
self._tcp_reconnect_sleep = 5
self._closing = False
if protocol == 'tcp':
self._tcp = True
@ -128,20 +132,25 @@ class StatsDCollector:
"""
sock = iostream.IOStream(socket.socket(
socket.AF_INET, socket.SOCK_STREAM, socket.IPPROTO_TCP))
sock.connect(self._address, self._tcp_on_connected)
sock.connect(self._address)
sock.set_close_callback(self._tcp_on_closed)
return sock
async def _tcp_on_closed(self):
"""Invoked when the socket is closed."""
if self._closing:
LOGGER.info('Statsd socket closed')
else:
LOGGER.warning('Not connected to statsd, connecting in %s seconds',
self._tcp_reconnect_sleep)
await asyncio.sleep(self._tcp_reconnect_sleep)
self._sock = self._tcp_socket()
def _tcp_on_connected(self):
"""Invoked when the IOStream is connected"""
LOGGER.debug('Connected to statsd at %s via TCP', self._address)
def close(self):
"""Gracefully close the socket."""
if not self._closing:
self._closing = True
self._sock.close()
def send(self, path, value, metric_type):
"""Send a metric to Statsd.
@ -205,16 +214,16 @@ def install(application, **kwargs):
:param tornado.web.Application application: the application to
install the collector into.
:param kwargs: keyword parameters to pass to the
:class:`StatsDCollector` initializer.
:class:`.StatsDCollector` initializer.
:returns: :data:`True` if the client was installed successfully,
or :data:`False` otherwise.
- **host** The StatsD host. If host is not specified, the
``STATSD_HOST`` environment variable, or default `127.0.0.1`,
will be pass into the :class:`StatsDCollector`.
will be pass into the :class:`.StatsDCollector`.
- **port** The StatsD port. If port is not specified, the
``STATSD_PORT`` environment variable, or default `8125`,
will be pass into the :class:`StatsDCollector`.
will be pass into the :class:`.StatsDCollector`.
- **namespace** The StatsD bucket to write metrics into.
"""
@ -232,3 +241,12 @@ def install(application, **kwargs):
setattr(application, 'statsd', StatsDCollector(**kwargs))
return True
def get_client(application):
"""Fetch the statsd client if it is installed.
:rtype: .StatsDCollector
"""
return getattr(application, 'statsd', None)

View file

@ -1,8 +1,7 @@
import asyncio
import itertools
import socket
import unittest
from unittest import mock
import unittest.mock
from tornado import iostream, testing, web
@ -48,7 +47,7 @@ class MisconfiguredStatsdMetricCollectionTests(testing.AsyncHTTPTestCase):
def test_bad_protocol_raises_ValueError(self):
with self.assertRaises(ValueError):
statsd.StatsDCollector(host='127.0.0.1',
port=8125,
port='8125',
protocol='bad_protocol')
@ -75,13 +74,13 @@ class TCPStatsdMetricCollectionTests(testing.AsyncHTTPTestCase):
'protocol': 'tcp',
'prepend_metric_type': True})
@mock.patch.object(iostream.IOStream, 'write')
@unittest.mock.patch.object(iostream.IOStream, 'write')
def test_write_not_executed_when_connection_is_closed(self, mock_write):
self.application.statsd._sock.close()
self.application.statsd.send('foo', 500, 'c')
mock_write.assert_not_called()
@mock.patch.object(iostream.IOStream, 'write')
@unittest.mock.patch.object(iostream.IOStream, 'write')
def test_expected_counters_data_written(self, mock_sock):
path = ('foo', 'bar')
value = 500
@ -94,7 +93,7 @@ class TCPStatsdMetricCollectionTests(testing.AsyncHTTPTestCase):
self.application.statsd.send(path, value, metric_type)
mock_sock.assert_called_once_with(expected.encode())
@mock.patch.object(iostream.IOStream, 'write')
@unittest.mock.patch.object(iostream.IOStream, 'write')
def test_expected_timers_data_written(self, mock_sock):
path = ('foo', 'bar')
value = 500
@ -156,6 +155,43 @@ class TCPStatsdMetricCollectionTests(testing.AsyncHTTPTestCase):
self.assertEqual(expected,
list(self.statsd.find_metrics(expected, 'ms'))[0][0])
def test_reconnect_logic(self):
self.application.statsd._tcp_reconnect_sleep = 0.05
self.application.statsd._sock.close()
asyncio.get_event_loop().run_until_complete(asyncio.sleep(0.075))
response = self.fetch('/status_code')
self.assertEqual(response.code, 200)
def test_that_mixin_works_without_client(self):
self.application.statsd.close()
delattr(self.application, 'statsd')
response = self.fetch('/', method='POST', body='')
self.assertEqual(response.code, 204)
def test_that_client_closes_socket(self):
response = self.fetch('/status_code')
self.assertEqual(response.code, 200)
self.application.statsd.close()
response = self.fetch('/status_code')
self.assertEqual(response.code, 200)
self.assertTrue(self.application.statsd._sock.closed())
def test_that_client_can_be_closed_multiple_times(self):
response = self.fetch('/status_code')
self.assertEqual(response.code, 200)
self.application.statsd.close()
response = self.fetch('/status_code')
self.assertEqual(response.code, 200)
self.assertTrue(self.application.statsd._sock.closed())
self.application.statsd.close()
response = self.fetch('/status_code')
self.assertEqual(response.code, 200)
self.assertTrue(self.application.statsd._sock.closed())
class TCPStatsdConfigurationTests(testing.AsyncHTTPTestCase):
@ -223,7 +259,7 @@ class UDPStatsdMetricCollectionTests(testing.AsyncHTTPTestCase):
self.statsd.close()
super().tearDown()
@mock.patch.object(socket.socket, 'sendto')
@unittest.mock.patch.object(socket.socket, 'sendto')
def test_expected_counters_data_written(self, mock_sock):
path = ('foo', 'bar')
value = 500
@ -238,7 +274,7 @@ class UDPStatsdMetricCollectionTests(testing.AsyncHTTPTestCase):
expected.encode(),
(self.statsd.sockaddr[0], self.statsd.sockaddr[1]))
@mock.patch.object(socket.socket, 'sendto')
@unittest.mock.patch.object(socket.socket, 'sendto')
def test_expected_timers_data_written(self, mock_sock):
path = ('foo', 'bar')
value = 500
@ -302,6 +338,13 @@ class UDPStatsdMetricCollectionTests(testing.AsyncHTTPTestCase):
self.assertEqual(expected,
list(self.statsd.find_metrics(expected, 'ms'))[0][0])
def test_that_mixin_works_without_client(self):
self.application.statsd.close()
delattr(self.application, 'statsd')
response = self.fetch('/', method='POST', body='')
self.assertEqual(response.code, 204)
class UDPStatsdConfigurationTests(testing.AsyncHTTPTestCase):

12
tox.ini
View file

@ -1,8 +1,18 @@
[tox]
envlist = py37
envlist = py37,tornado5,tornado6
toxworkdir = build/tox
skip_missing_interpreters = True
[testenv]
deps = -r requires/testing.txt
commands = nosetests
[testenv:tornado5]
deps =
tornado>=5,<6
-r requires/testing.txt
[testenv:tornado6]
deps =
tornado>=6,<7
-r requires/testing.txt