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

Commit

Permalink
feat: make gcm calls use async callbacks
Browse files Browse the repository at this point in the history
Closes #1291
  • Loading branch information
jrconlin committed Oct 1, 2018
1 parent a569714 commit 00432dc
Show file tree
Hide file tree
Showing 7 changed files with 299 additions and 148 deletions.
54 changes: 31 additions & 23 deletions autopush/router/gcm.py
Original file line number Diff line number Diff line change
@@ -1,9 +1,8 @@
"""GCM Router"""
from typing import Any # noqa

from requests.exceptions import ConnectionError, Timeout
from twisted.internet.threads import deferToThread
from twisted.logger import Logger
from twisted.internet.error import ConnectError, TimeoutError

from autopush.exceptions import RouterException
from autopush.metrics import make_tags
Expand Down Expand Up @@ -36,7 +35,8 @@ def __init__(self, conf, router_conf, metrics):
for sid in router_conf.get("senderIDs"):
auth = router_conf.get("senderIDs").get(sid).get("auth")
self.senderIDs[sid] = auth
self.gcm[sid] = gcmclient.GCM(auth, timeout=timeout)
self.gcm[sid] = gcmclient.GCM(auth, timeout=timeout,
logger=self.log)
self._base_tags = ["platform:gcm"]
self.log.debug("Starting GCM router...")

Expand Down Expand Up @@ -69,7 +69,7 @@ def register(self, uaid, router_data, app_id, *args, **kwargs):
def route_notification(self, notification, uaid_data):
"""Start the GCM notification routing, returns a deferred"""
# Kick the entire notification routing off to a thread
return deferToThread(self._route, notification, uaid_data)
return self._route(notification, uaid_data)

def _route(self, notification, uaid_data):
"""Blocking GCM call to route the notification"""
Expand Down Expand Up @@ -112,52 +112,60 @@ def _route(self, notification, uaid_data):
)
try:
gcm = self.gcm[router_data['creds']['senderID']]
result = gcm.send(payload)
except RouterException:
raise # pragma nocover
d = gcm.send(payload)
d.addCallback(
self._process_reply,
uaid_data,
router_ttl,
notification)

d.addErrback(
self._process_error
)
return d
except KeyError:
self.log.critical("Missing GCM bridge credentials")
raise RouterException("Server error", status_code=500,
errno=900)
except gcmclient.GCMAuthenticationError as e:
self.log.error("GCM Authentication Error: %s" % e)

def _process_error(self, failure):
err = failure.value
if isinstance(err, gcmclient.GCMAuthenticationError):
self.log.error("GCM Authentication Error: %s" % err)
raise RouterException("Server error", status_code=500,
errno=901)
except ConnectionError as e:
self.log.warn("GCM Unavailable: %s" % e)
if isinstance(err, TimeoutError):
self.log.warn("GCM Timeout: %s" % err)
self.metrics.increment("notification.bridge.error",
tags=make_tags(
self._base_tags,
reason="connection_unavailable"))
reason="timeout"))
raise RouterException("Server error", status_code=502,
errno=902,
errno=903,
log_exception=False)
except Timeout as e:
self.log.warn("GCM Timeout: %s" % e)
if isinstance(err, ConnectError):
self.log.warn("GCM Unavailable: %s" % err)
self.metrics.increment("notification.bridge.error",
tags=make_tags(
self._base_tags,
reason="timeout"))
reason="connection_unavailable"))
raise RouterException("Server error", status_code=502,
errno=903,
errno=902,
log_exception=False)
except Exception as e:
self.log.error("Unhandled exception in GCM Routing: %s" % e)
raise RouterException("Server error", status_code=500)
return self._process_reply(result, uaid_data, ttl=router_ttl,
notification=notification)
return failure

def _error(self, err, status, **kwargs):
"""Error handler that raises the RouterException"""
self.log.debug(err, **kwargs)
return RouterException(err, status_code=status, response_body=err,
**kwargs)

def _process_reply(self, reply, uaid_data, ttl, notification):
def _process_reply(self, gcm, uaid_data, ttl, notification):
"""Process GCM send reply"""
# acks:
# for reg_id, msg_id in reply.success.items():
# updates
reply = gcm.response
for old_id, new_id in reply.canonicals.items():
self.log.debug("GCM id changed : {old} => {new}",
old=old_id, new=new_id)
Expand Down
78 changes: 53 additions & 25 deletions autopush/router/gcmclient.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,9 @@
import json

import requests
import treq
from twisted.web.http_headers import Headers
from twisted.logger import Logger
from twisted.internet.error import ConnectError

from autopush.exceptions import RouterException

Expand All @@ -12,7 +15,7 @@ class GCMAuthenticationError(Exception):
class Result(object):
"""Abstraction object for GCM response"""

def __init__(self, message, response):
def __init__(self, response, message):
"""Process GCM message and response into abstracted object
:param message: Message payload
Expand All @@ -30,14 +33,12 @@ def __init__(self, message, response):
self.message = message
self.retry_message = None

self.retry_after = response.headers.get('Retry-After', None)
self.retry_after = (
response.headers.getRawHeaders('Retry-After') or [None])[0]

if response.status_code != 200:
self.retry_message = message
else:
self._parse_response(message, response.content)

def _parse_response(self, message, content):
def parse_response(self, content, code, message):
if code in (400, 401, 404):
raise RouterException(content)
data = json.loads(content)
if not data.get('results'):
raise RouterException("Recv'd invalid response from GCM")
Expand All @@ -54,6 +55,7 @@ def _parse_response(self, message, content):
self.not_registered.append(reg_id)
else:
self.failed[reg_id] = res['error']
return self


class JSONMessage(object):
Expand Down Expand Up @@ -124,9 +126,36 @@ def __init__(self,
self._endpoint = "https://{}".format(endpoint)
self._api_key = api_key
self.metrics = metrics
self.log = logger
self.log = logger or Logger()
self._options = options
self._sender = requests.post
self._sender = treq.request

def _set_response(self, response):
self.response = response
return self

def process(self, response, payload):
if response.code == 401:
raise GCMAuthenticationError("Authentication Error")

result = Result(response, payload)

if 500 <= response.code <= 599:
result.retry_message = payload
return self._set_response(result)

# Fetch the content body
d = response.text()
d.addCallback(result.parse_response, response.code, payload)
d.addCallback(self._set_response)
return d

def error(self, failure):
if isinstance(failure.value, GCMAuthenticationError) or \
isinstance(failure.value, ConnectError):
raise failure.value
self.log.error("GCMClient failure: {}".format(failure.value))
raise RouterException("Server error: {}".format(failure.value))

def send(self, payload):
"""Send a payload to GCM
Expand All @@ -136,23 +165,22 @@ def send(self, payload):
:return: Result
"""
headers = {
'Content-Type': 'application/json',
'Authorization': 'key={}'.format(self._api_key),
}
headers = Headers({
'Content-Type': ['application/json'],
'Authorization': ['key={}'.format(self._api_key)],
})

response = self._sender(
if 'timeout' not in self._options:
self._options['timeout'] = 3

d = self._sender(
method="POST",
url=self._endpoint,
headers=headers,
data=json.dumps(payload.payload),
**self._options
)

if response.status_code in (400, 404):
raise RouterException(response.content)

if response.status_code == 401:
raise GCMAuthenticationError("Authentication Error")

if response.status_code == 200 or (500 <= response.status_code <= 599):
return Result(payload, response)
# handle the immediate response (which contains no body)
d.addCallback(self.process, payload)
d.addErrback(self.error)
return d
Loading

0 comments on commit 00432dc

Please sign in to comment.