From 377d0cbe52dd977fd8d3ff2c83998f2db6818555 Mon Sep 17 00:00:00 2001 From: jrconlin Date: Fri, 7 Aug 2020 00:51:33 +0000 Subject: [PATCH 1/3] bug: report details of an FCM "INVALID_ARGUMENT" error Issue #1373 --- autopush/router/fcmv1client.py | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/autopush/router/fcmv1client.py b/autopush/router/fcmv1client.py index e5b16ea0..57992303 100644 --- a/autopush/router/fcmv1client.py +++ b/autopush/router/fcmv1client.py @@ -63,8 +63,11 @@ def parse_response(self, content): err = data.get("error") if err.get("status") == "NOT_FOUND": raise FCMNotFoundError("FCM recipient no longer available") - raise RouterException("{}: {}".format(err.get("status"), - err.get("message"))) + raise RouterException( + "{}: {} {}".format( + err.get("status"), + err.get("message"), + err.get("details", "No Details"))) if "name" in data: self.success = 1 except (TypeError, ValueError, KeyError, AttributeError): From e13b575c8731bde712aa8b4a120e126942b974c5 Mon Sep 17 00:00:00 2001 From: jrconlin Date: Fri, 7 Aug 2020 01:05:31 +0000 Subject: [PATCH 2/3] f flake8 --- autopush/router/fcmv1client.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/autopush/router/fcmv1client.py b/autopush/router/fcmv1client.py index 57992303..2729124b 100644 --- a/autopush/router/fcmv1client.py +++ b/autopush/router/fcmv1client.py @@ -65,9 +65,9 @@ def parse_response(self, content): raise FCMNotFoundError("FCM recipient no longer available") raise RouterException( "{}: {} {}".format( - err.get("status"), - err.get("message"), - err.get("details", "No Details"))) + err.get("status"), + err.get("message"), + err.get("details", "No Details"))) if "name" in data: self.success = 1 except (TypeError, ValueError, KeyError, AttributeError): From 6499214522fab090417a0fdb465f0e2dcd228ded Mon Sep 17 00:00:00 2001 From: jrconlin Date: Fri, 7 Aug 2020 19:24:43 +0000 Subject: [PATCH 3/3] f Check message length for fcm * Ensure that outbound messages cap at 4096 bytes for the payload * Try to be helpful about message lengths. --- autopush/router/apnsrouter.py | 17 +++++----- autopush/router/fcm_v1.py | 61 +++++++++++++++++++++++++---------- autopush/tests/test_router.py | 6 ++-- 3 files changed, 57 insertions(+), 27 deletions(-) diff --git a/autopush/router/apnsrouter.py b/autopush/router/apnsrouter.py index 9be65ae1..60ec6baa 100644 --- a/autopush/router/apnsrouter.py +++ b/autopush/router/apnsrouter.py @@ -128,15 +128,16 @@ def _route(self, notification, router_data): "ver": notification.version, } if notification.data: + payload["con"] = notification.headers.get( + "content-encoding", notification.headers.get("encoding")) + if payload["con"] != "aes128gcm": + if "encryption" in notification.headers: + payload["enc"] = notification.headers["encryption"] + if "crypto_key" in notification.headers: + payload["cryptokey"] = notification.headers["crypto_key"] + elif "encryption_key" in notification.headers: + payload["enckey"] = notification.headers["encryption_key"] payload["body"] = notification.data - payload["con"] = notification.headers["encoding"] - - if "encryption" in notification.headers: - payload["enc"] = notification.headers["encryption"] - if "crypto_key" in notification.headers: - payload["cryptokey"] = notification.headers["crypto_key"] - elif "encryption_key" in notification.headers: - payload["enckey"] = notification.headers["encryption_key"] payload['aps'] = router_data.get('aps', { "mutable-content": 1, "alert": { diff --git a/autopush/router/fcm_v1.py b/autopush/router/fcm_v1.py index 0b1a5374..5f685060 100644 --- a/autopush/router/fcm_v1.py +++ b/autopush/router/fcm_v1.py @@ -1,4 +1,7 @@ """FCM v1 HTTP Router""" +import json +from math import ceil + from typing import Any # noqa from twisted.internet.error import ConnectError, TimeoutError @@ -15,6 +18,13 @@ ) from autopush.types import JSONDict # noqa +# the universal default for this is 4096, which is far too +# large for FCM. The final payload size of the encoded data, plus +# encryption headers must fit in the 4096 byte FCM payload. +# Since the body is re-encoded base64, we reduce the message +# size accordingly +MAX_FCM_DATA = 3015 + class FCMv1Router(FCMRouter): """FCM v1 HTTP Router Implementation @@ -81,25 +91,42 @@ def _route(self, notification, router_data): # Payload data is optional. The endpoint handler validates that the # correct encryption headers are included with the data. if notification.data: - mdata = self.router_conf.get('max_data', 4096) + data['con'] = notification.headers.get('encoding') + mdata = self.router_conf.get('max_data', MAX_FCM_DATA) + if data['con'] != "aes128gcm": + # aes128gcm does not include headers, so they get more data. + if 'encryption' in notification.headers: + data['enc'] = notification.headers['encryption'] + if 'crypto_key' in notification.headers: + data['cryptokey'] = notification.headers['crypto_key'] + elif 'encryption_key' in notification.headers: + data['enckey'] = notification.headers['encryption_key'] + data["body"] = "" + mdata = mdata - len(json.dumps(data)) + data['body'] = notification.data if notification.data_length > mdata: - raise self._error("This message is intended for a " + - "constrained device and is limited " + - "to 3070 bytes. Converted buffer too " + - "long by %d bytes" % - (notification.data_length - mdata), + # take a guess at about how long the decoded message buffer + # needs to be. + suggest_length = int( + ceil((notification.data_length - mdata) / 1.3)) + raise self._error("This message is intended for a " + "constrained device and is limited " + "to {} bytes. Message too " + "long by about {} bytes".format( + mdata, suggest_length), 413, errno=104, log_exception=False) - - data['body'] = notification.data - data['con'] = notification.headers['encoding'] - - if 'encryption' in notification.headers: - data['enc'] = notification.headers['encryption'] - if 'crypto_key' in notification.headers: - data['cryptokey'] = notification.headers['crypto_key'] - elif 'encryption_key' in notification.headers: - data['enckey'] = notification.headers['encryption_key'] - + # check the size of the outbound message data, again. + payload_size = len(json.dumps(data)) + # 4096 is the hard limit for FCM payloads. Trap just in case + # our math was wrong. + if payload_size > 4096: + raise self._error( + "Final composed message payload too long for recipient: " + "{} bytes. Please try a shorter message.".format( + payload_size, + 4096 + ), + 413, errno=104, log_exception=False) # registration_ids are the FCM instance tokens (specified during # registration. router_ttl = min(self.MAX_TTL, diff --git a/autopush/tests/test_router.py b/autopush/tests/test_router.py index 069d5e9e..942b8f67 100644 --- a/autopush/tests/test_router.py +++ b/autopush/tests/test_router.py @@ -761,7 +761,7 @@ def setUp(self): hostname="localhost", statsd_host=None, ) - self.fcm_config = {'max_data': 32, + self.fcm_config = {'max_data': 120, 'ttl': 60, 'version': 1, 'dryrun': False, @@ -940,10 +940,12 @@ def test_long_data(self): bad_notif = WebPushNotification( uaid=uuid.UUID(dummy_uaid), channel_id=uuid.UUID(dummy_chid), - data="\x01abcdefghijklmnopqrstuvwxyz0123456789", + data="\x01" + "a" * 120, headers=self.headers, ttl=200 ) + # fix up headers since we're calling route directly. + bad_notif.cleanup_headers() self._set_content() with pytest.raises(RouterException) as ex: