Skip to content
This repository has been archived by the owner on Jul 13, 2023. It is now read-only.

Commit

Permalink
feat: Add request timing w/validation breakdown.
Browse files Browse the repository at this point in the history
Add's full request timing with validation and routing break-outs. Also logs the request
and validation timing even if validation fails or routing otherwise errors out.

Closes #758
  • Loading branch information
bbangert committed Dec 31, 2016
1 parent a9cffbb commit c460b92
Show file tree
Hide file tree
Showing 7 changed files with 62 additions and 13 deletions.
9 changes: 8 additions & 1 deletion autopush/logging.py
Original file line number Diff line number Diff line change
Expand Up @@ -165,8 +165,15 @@ def to_fields(kv):
if ci and isinstance(ci, dict):
msg['Fields'].update(
to_fields(ci.iteritems()))
# Add the nicely formatted message

# flatten timings into Fields
ti = event.get('timings')
if ti and isinstance(ti, dict):
msg["Fields"].update(
to_fields(ti.iteritems())
)

# Add the nicely formatted message
msg["Fields"]["message"] = formatEvent(event)
return json.dumps(msg, skipkeys=True) + "\n"

Expand Down
2 changes: 1 addition & 1 deletion autopush/router/simple.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,10 +6,10 @@
"""
import json
import requests
from urllib import urlencode
from StringIO import StringIO

import requests
from boto.dynamodb2.exceptions import (
ItemNotFound,
ProvisionedThroughputExceededException,
Expand Down
8 changes: 6 additions & 2 deletions autopush/tests/test_logging.py
Original file line number Diff line number Diff line change
Expand Up @@ -50,10 +50,12 @@ def test_sentry_logging(self):
pl = PushLogger.setup_logging("Autopush", sentry_dsn=True)
pl._output = out
_client_info = dict(key='value')
_timings = dict(key2='value')

log.failure(format="error",
failure=failure.Failure(Exception("eek")),
client_info=_client_info)
client_info=_client_info,
timings=_timings)
self.flushLoggedErrors()
d = Deferred()

Expand All @@ -70,7 +72,9 @@ def check():
# Check that the json written actually contains the client info
# collapsed up into 'Fields'.
out.seek(0)
eq_(json.loads(out.readline())['Fields']['key'], 'value')
payload = json.loads(out.readline())
eq_(payload['Fields']['key'], 'value')
eq_(payload['Fields']['key2'], 'value')
self._port.stopListening()
pl.stop()
d.callback(True)
Expand Down
8 changes: 6 additions & 2 deletions autopush/tests/test_web_validation.py
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,7 @@ def _write_validation_err(rh, errors):
app.ui_modules = dict()
app.ui_methods = dict()
vr = ValidateRequest(app, request)
vr._timings = dict()
vr.ap_settings = Mock()
return vr

Expand All @@ -97,17 +98,19 @@ def test_validate_invalid_schema(self):
eq_(d, {})

def test_call_func_no_error(self):
start_time = time.time()
mock_func = Mock()
tv, rh = self._make_full()
result = tv._validate_request(rh)
tv._call_func(result, mock_func, rh)
tv._call_func(result, mock_func, rh, start_time)
mock_func.assert_called()

def test_call_func_error(self):
start_time = time.time()
mock_func = Mock()
tv, rh = self._make_full(schema=InvalidSchema)
result = tv._validate_request(rh)
tv._call_func(result, mock_func, rh)
tv._call_func(result, mock_func, rh, start_time)
self._mock_errors.assert_called()
eq_(len(mock_func.mock_calls), 0)

Expand All @@ -127,6 +130,7 @@ def get(self):
app.ui_modules = dict()
app.ui_methods = dict()
vr = AHandler(app, req)
vr._timings = dict()
d = Deferred()
vr.finish = lambda: d.callback(True)
vr.write = Mock()
Expand Down
40 changes: 36 additions & 4 deletions autopush/web/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,10 @@
from attr import attrs, attrib
from boto.dynamodb2.exceptions import ProvisionedThroughputExceededException
from boto.exception import BotoServerError
from marshmallow.schema import UnmarshalResult # noqa
from twisted.internet.threads import deferToThread
from twisted.logger import Logger
from twisted.python.failure import Failure # noqa

from autopush.base import BaseHandler
from autopush.exceptions import InvalidRequest, RouterException
Expand Down Expand Up @@ -43,6 +45,7 @@ def __init__(self, schema):
self.schema = schema

def _validate_request(self, request_handler):
# type: (BaseWebHandler) -> UnmarshalResult
"""Validates a schema_class against a cyclone request"""
data = {
"headers": request_handler.request.headers,
Expand All @@ -56,22 +59,32 @@ def _validate_request(self, request_handler):
schema.context["log"] = self.log
return schema.load(data)

def _call_func(self, result, func, request_handler, *args, **kwargs):
def _call_func(self, result, func, request_handler, start_time, *args,
**kwargs):
output, errors = result
request_handler._timings["validation_time"] = time.time() - start_time
if errors:
request_handler._write_validation_err(errors)
else:
request_handler.valid_input = output
return func(request_handler, *args, **kwargs)

def _track_error_timing(self, failure, request_handler, start_time):
# type: (Failure, BaseWebHandler, float) -> None
"""Track the validation timing for the error"""
request_handler._timings["validation_time"] = time.time() - start_time
failure.raiseException()

def _decorator(self, func):
@wraps(func)
def wrapper(request_handler, *args, **kwargs):
start_time = time.time()
# Wrap the handler in @cyclone.web.synchronous
request_handler._auto_finish = False
d = deferToThread(self._validate_request, request_handler)
d.addCallback(self._call_func, func, request_handler, *args,
**kwargs)
d.addCallback(self._call_func, func, request_handler, start_time,
*args, **kwargs)
d.addErrback(self._track_error_timing, request_handler, start_time)
d.addErrback(request_handler._overload_err)
d.addErrback(request_handler._boto_err)
d.addErrback(request_handler._validation_err)
Expand Down Expand Up @@ -124,9 +137,10 @@ class BaseWebHandler(BaseHandler):
def initialize(self, ap_settings):
"""Setup basic aliases and attributes"""
super(BaseWebHandler, self).initialize(ap_settings)
self.start_time = time.time()
self.metrics = ap_settings.metrics
self._base_tags = {}
self._start_time = time.time()
self._timings = {}

def prepare(self):
"""Common request preparation"""
Expand Down Expand Up @@ -171,6 +185,7 @@ def _write_response(self, status_code, errno, message=None, error=None,
if headers:
for header in headers.keys():
self.set_header(header, headers.get(header))
self._track_timing()
self.finish()

def _validation_err(self, fail):
Expand Down Expand Up @@ -224,6 +239,7 @@ def _router_response(self, response):
if 200 <= response.status_code < 300:
self.set_status(response.status_code, reason=None)
self.write(response.response_body)
self._track_timing(status_code=response.logged_status)
self.finish()
else:
self._write_response(
Expand Down Expand Up @@ -264,6 +280,7 @@ def _write_validation_err(self, errors):
errors=errors
)
self.write(json.dumps(error_data))
self._track_timing()
self.finish()

def _db_error_handling(self, d):
Expand All @@ -273,3 +290,18 @@ def _db_error_handling(self, d):
d.addErrback(self._boto_err)
d.addErrback(self._response_err)
return d

#############################################################
# Utility Methods
#############################################################
def _track_timing(self, status_code=None):
"""Logs out the request timing tracking stats
Note: The status code should be set before calling this function or
passed in.
"""
status_code = status_code or self.get_status()
self._timings["request_time"] = time.time() - self._start_time
self.log.info("Request timings", client_info=self._client_info,
timings=self._timings, status_code=status_code)
2 changes: 1 addition & 1 deletion autopush/web/simplepush.py
Original file line number Diff line number Diff line change
Expand Up @@ -131,7 +131,7 @@ def _router_completed(self, response, uaid_data, warning=""):
elif response.status_code == 202 or response.logged_status == 202:
self.log.info(format="Router miss, message stored.",
client_info=self._client_info)
time_diff = time.time() - self.start_time
time_diff = time.time() - self._start_time
self.metrics.timing("updates.handled", duration=time_diff)
response.response_body = (
response.response_body + " " + warning).strip()
Expand Down
6 changes: 4 additions & 2 deletions autopush/web/webpush.py
Original file line number Diff line number Diff line change
Expand Up @@ -348,6 +348,7 @@ def post(self, *args, **kwargs):
self._client_info["message_size"] = len(notification.data or "")
self._client_info["ttl"] = notification.ttl
self._client_info["version"] = notification.version
self._router_time = time.time()
d = Deferred()
d.addCallback(router.route_notification, user_data)
d.addCallback(self._router_completed, user_data, "")
Expand All @@ -359,9 +360,10 @@ def post(self, *args, **kwargs):

def _router_completed(self, response, uaid_data, warning=""):
"""Called after router has completed successfully"""
# Log the time taken for routing
self._timings["route_time"] = time.time() - self._router_time
# Were we told to update the router data?
time_diff = time.time() - self.start_time
self._client_info["route_time"] = time_diff
time_diff = time.time() - self._start_time
if response.router_data is not None:
if not response.router_data:
# An empty router_data object indicates that the record should
Expand Down

0 comments on commit c460b92

Please sign in to comment.