diff --git a/README.md b/README.md
index 33524ee1..27a42315 100644
--- a/README.md
+++ b/README.md
@@ -108,7 +108,7 @@ The table below identifies the services this tool supports and some example serv
| [Pushy](https://github.com/caronc/apprise/wiki/Notify_pushy) | pushy:// | (TCP) 443 | pushy://apikey/DEVICE
pushy://apikey/DEVICE1/DEVICE2/DEVICEN
pushy://apikey/TOPIC
pushy://apikey/TOPIC1/TOPIC2/TOPICN
| [PushDeer](https://github.com/caronc/apprise/wiki/Notify_pushdeer) | pushdeer:// or pushdeers:// | (TCP) 80 or 443 | pushdeer://pushKey
pushdeer://hostname/pushKey
pushdeer://hostname:port/pushKey
| [Reddit](https://github.com/caronc/apprise/wiki/Notify_reddit) | reddit:// | (TCP) 443 | reddit://user:password@app_id/app_secret/subreddit
reddit://user:password@app_id/app_secret/sub1/sub2/subN
-| [Revolt](https://github.com/caronc/apprise/wiki/Notify_Revolt) | revolt:// | (TCP) 443 | revolt://bot_token/channel_id |
+| [Revolt](https://github.com/caronc/apprise/wiki/Notify_Revolt) | revolt:// | (TCP) 443 | revolt://bottoken/ChannelID
revolt://bottoken/ChannelID1/ChannelID2/ChannelIDN |
| [Rocket.Chat](https://github.com/caronc/apprise/wiki/Notify_rocketchat) | rocket:// or rockets:// | (TCP) 80 or 443 | rocket://user:password@hostname/RoomID/Channel
rockets://user:password@hostname:443/#Channel1/#Channel1/RoomID
rocket://user:password@hostname/#Channel
rocket://webhook@hostname
rockets://webhook@hostname/@User/#Channel
| [RSyslog](https://github.com/caronc/apprise/wiki/Notify_rsyslog) | rsyslog:// | (UDP) 514 | rsyslog://hostname
rsyslog://hostname/Facility
| [Ryver](https://github.com/caronc/apprise/wiki/Notify_ryver) | ryver:// | (TCP) 443 | ryver://Organization/Token
ryver://botname@Organization/Token
diff --git a/apprise/URLBase.py b/apprise/URLBase.py
index c9f5877f..2467a4c1 100644
--- a/apprise/URLBase.py
+++ b/apprise/URLBase.py
@@ -28,7 +28,7 @@
import re
from .logger import logger
-from time import sleep
+import time
from datetime import datetime
from xml.sax.saxutils import escape as sax_escape
@@ -298,12 +298,12 @@ class URLBase:
if wait is not None:
self.logger.debug('Throttling forced for {}s...'.format(wait))
- sleep(wait)
+ time.sleep(wait)
elif elapsed < self.request_rate_per_sec:
self.logger.debug('Throttling for {}s...'.format(
self.request_rate_per_sec - elapsed))
- sleep(self.request_rate_per_sec - elapsed)
+ time.sleep(self.request_rate_per_sec - elapsed)
# Update our timestamp before we leave
self._last_io_datetime = datetime.now()
diff --git a/apprise/plugins/NotifyRevolt.py b/apprise/plugins/NotifyRevolt.py
index 24cfeee6..ae0a43b1 100644
--- a/apprise/plugins/NotifyRevolt.py
+++ b/apprise/plugins/NotifyRevolt.py
@@ -37,7 +37,7 @@
#
import requests
-from json import dumps
+from json import dumps, loads
from datetime import timedelta
from datetime import datetime
from datetime import timezone
@@ -46,8 +46,8 @@ from .NotifyBase import NotifyBase
from ..common import NotifyImageSize
from ..common import NotifyFormat
from ..common import NotifyType
-from ..utils import parse_bool
from ..utils import validate_regex
+from ..utils import parse_list
from ..AppriseLocale import gettext_lazy as _
@@ -60,7 +60,7 @@ class NotifyRevolt(NotifyBase):
service_name = 'Revolt'
# The services URL
- service_url = 'https://api.revolt.chat/'
+ service_url = 'https://revolt.chat/'
# The default secure protocol
secure_protocol = 'revolt'
@@ -71,7 +71,7 @@ class NotifyRevolt(NotifyBase):
# Revolt Channel Message
notify_url = 'https://api.revolt.chat/'
- # Revolt supports attachments but don't implemenet for now
+ # Revolt supports attachments but doesn't support it here (for now)
attachment_support = False
# Allows the user to specify the NotifyImageSize object
@@ -85,8 +85,8 @@ class NotifyRevolt(NotifyBase):
# still allow to make.
request_rate_per_sec = 3
- # Taken right from google.auth.helpers:
- clock_skew = timedelta(seconds=10)
+ # Safety net
+ clock_skew = timedelta(seconds=2)
# The maximum allowable characters allowed in the body per message
body_maxlen = 2000
@@ -96,7 +96,7 @@ class NotifyRevolt(NotifyBase):
# Define object templates
templates = (
- '{schema}://{bot_token}/{channel_id}',
+ '{schema}://{bot_token}/{targets}',
)
# Defile out template tokens
@@ -107,39 +107,44 @@ class NotifyRevolt(NotifyBase):
'private': True,
'required': True,
},
- 'channel_id': {
- 'name': _('Channel Id'),
+ 'target_channel': {
+ 'name': _('Channel ID'),
'type': 'string',
+ 'map_to': 'targets',
+ 'regex': (r'^[a-z0-9_-]+$', 'i'),
'private': True,
'required': True,
},
+ 'targets': {
+ 'name': _('Targets'),
+ 'type': 'list:string',
+ },
})
# Define our template arguments
template_args = dict(NotifyBase.template_args, **{
- 'channel_id': {
- 'alias_of': 'channel_id',
+ 'channel': {
+ 'alias_of': 'targets',
},
'bot_token': {
'alias_of': 'bot_token',
},
- 'embed_img': {
- 'name': _('Embed Image Url'),
+ 'icon_url': {
+ 'name': _('Icon URL'),
'type': 'string'
},
- 'embed_url': {
- 'name': _('Embed Url'),
- 'type': 'string'
+ 'url': {
+ 'name': _('Embed URL'),
+ 'type': 'string',
+ 'map_to': 'link',
+ },
+ 'to': {
+ 'alias_of': 'targets',
},
- 'custom_img': {
- 'name': _('Custom Embed Url'),
- 'type': 'bool',
- 'default': False
- }
})
- def __init__(self, bot_token, channel_id, embed_img=None, embed_url=None,
- custom_img=None, **kwargs):
+ def __init__(self, bot_token, targets, icon_url=None, link=None,
+ **kwargs):
super().__init__(**kwargs)
# Bot Token
@@ -150,24 +155,27 @@ class NotifyRevolt(NotifyBase):
self.logger.warning(msg)
raise TypeError(msg)
- # Channel Id
- self.channel_id = validate_regex(channel_id)
- if not self.channel_id:
- msg = 'An invalid Revolt Channel Id' \
- '({}) was specified.'.format(channel_id)
- self.logger.warning(msg)
- raise TypeError(msg)
+ # Parse our Channel IDs
+ self.targets = []
+ for target in parse_list(targets):
+ results = validate_regex(
+ target, *self.template_tokens['target_channel']['regex'])
- # Use custom image for embed image
- self.custom_img = parse_bool(custom_img) \
- if custom_img is not None \
- else self.template_args['custom_img']['default']
+ if not results:
+ self.logger.warning(
+ 'Dropped invalid Revolt channel ({}) specified.'
+ .format(target),
+ )
+ continue
+
+ # Add our target
+ self.targets.append(target)
# Image for Embed
- self.embed_img = embed_img
+ self.icon_url = icon_url
# Url for embed title
- self.embed_url = embed_url
+ self.link = link
# For Tracking Purposes
self.ratelimit_reset = datetime.now(timezone.utc).replace(tzinfo=None)
@@ -183,44 +191,47 @@ class NotifyRevolt(NotifyBase):
"""
+ if len(self.targets) == 0:
+ self.logger.warning('There were not Revolt channels to notify.')
+ return False
+
payload = {}
# Acquire image_url
- image_url = self.image_url(notify_type)
+ image_url = self.icon_url \
+ if self.icon_url else self.image_url(notify_type)
- if self.custom_img and (image_url or self.embed_url):
- image_url = self.embed_url if self.embed_url else image_url
+ if self.notify_format == NotifyFormat.MARKDOWN:
+ payload['embeds'] = [{
+ 'title': None if not title else title[0:self.title_maxlen],
+ 'description': body,
- if body:
- if self.notify_format == NotifyFormat.MARKDOWN:
- if len(title) > 100:
- msg = 'Title length must be less than 100 when ' \
- 'embeds are enabled (is %s)' % len(title)
- self.logger.warning(msg)
- title = title[0:100]
- payload['embeds'] = [{
- 'title': title,
- 'description': body,
+ # Our color associated with our notification
+ 'colour': self.color(notify_type),
+ 'replies': None
+ }]
- # Our color associated with our notification
- 'colour': self.color(notify_type, int)
- }]
+ if image_url:
+ payload['embeds'][0]['icon_url'] = image_url
- if self.embed_img:
- payload['embeds'][0]['icon_url'] = image_url
+ if self.link:
+ payload['embeds'][0]['url'] = self.link
- if self.embed_url:
- payload['embeds'][0]['url'] = self.embed_url
- else:
- payload['content'] = \
- body if not title else "{}\n{}".format(title, body)
+ else:
+ payload['content'] = \
+ body if not title else "{}\n{}".format(title, body)
- if not self._send(payload):
+ has_error = False
+ channel_ids = list(self.targets)
+ for channel_id in channel_ids:
+ postokay, response = self._send(payload, channel_id)
+ if not postokay:
# Failed to send message
- return False
- return True
+ has_error = True
- def _send(self, payload, rate_limit=1, **kwargs):
+ return not has_error
+
+ def _send(self, payload, channel_id, retries=1, **kwargs):
"""
Wrapper to the requests (post) object
@@ -229,12 +240,13 @@ class NotifyRevolt(NotifyBase):
headers = {
'User-Agent': self.app_id,
'X-Bot-Token': self.bot_token,
- 'Content-Type': 'application/json; charset=utf-8'
+ 'Content-Type': 'application/json; charset=utf-8',
+ 'Accept': 'application/json; charset=utf-8',
}
notify_url = '{0}channels/{1}/messages'.format(
self.notify_url,
- self.channel_id
+ channel_id
)
self.logger.debug('Revolt POST URL: %s (cert_verify=%r)' % (
@@ -245,20 +257,22 @@ class NotifyRevolt(NotifyBase):
# By default set wait to None
wait = None
+ now = datetime.now(timezone.utc).replace(tzinfo=None)
if self.ratelimit_remaining <= 0.0:
# Determine how long we should wait for or if we should wait at
# all. This isn't fool-proof because we can't be sure the client
# time (calling this script) is completely synced up with the
# Discord server. One would hope we're on NTP and our clocks are
# the same allowing this to role smoothly:
-
- now = datetime.now(timezone.utc).replace(tzinfo=None)
if now < self.ratelimit_reset:
# We need to throttle for the difference in seconds
wait = abs(
(self.ratelimit_reset - now + self.clock_skew)
.total_seconds())
+ # Default content response object
+ content = {}
+
# Always call throttle before any remote server i/o is made;
self.throttle(wait=wait)
@@ -271,15 +285,23 @@ class NotifyRevolt(NotifyBase):
timeout=self.request_timeout
)
+ try:
+ content = loads(r.content)
+
+ except (AttributeError, TypeError, ValueError):
+ # ValueError = r.content is Unparsable
+ # TypeError = r.content is None
+ # AttributeError = r is None
+ content = {}
+
# Handle rate limiting (if specified)
try:
# Store our rate limiting (if provided)
self.ratelimit_remaining = \
- float(r.headers.get(
- 'X-RateLimit-Remaining'))
- self.ratelimit_reset = datetime.fromtimestamp(
- int(r.headers.get('X-RateLimit-Reset')),
- timezone.utc).replace(tzinfo=None)
+ int(r.headers.get('X-RateLimit-Remaining'))
+ self.ratelimit_reset = \
+ now + timedelta(seconds=(int(
+ r.headers.get('X-RateLimit-Reset-After')) / 1000))
except (TypeError, ValueError):
# This is returned if we could not retrieve this
@@ -289,23 +311,27 @@ class NotifyRevolt(NotifyBase):
if r.status_code not in (
requests.codes.ok, requests.codes.no_content):
+ # Some details to debug by
+ self.logger.debug('Response Details:\r\n{}'.format(
+ content if content else r.content))
+
# We had a problem
status_str = \
NotifyBase.http_response_code_lookup(r.status_code)
+ self.logger.warning(
+ 'Revolt request limit reached; '
+ 'instructed to throttle for %.3fs',
+ abs((self.ratelimit_reset - now + self.clock_skew)
+ .total_seconds()))
+
if r.status_code == requests.codes.too_many_requests \
- and rate_limit > 0:
+ and retries > 0:
- # handle rate limiting
- self.logger.warning(
- 'Revolt rate limiting in effect; '
- 'blocking for %.2f second(s)',
- self.ratelimit_remaining)
-
- # Try one more time before failing
+ # Try again
return self._send(
- payload=payload,
- rate_limit=rate_limit - 1, **kwargs)
+ payload=payload, channel_id=channel_id,
+ retries=retries - 1, **kwargs)
self.logger.warning(
'Failed to send to Revolt notification: '
@@ -314,10 +340,8 @@ class NotifyRevolt(NotifyBase):
', ' if status_str else '',
r.status_code))
- self.logger.debug('Response Details:\r\n{}'.format(r.content))
-
# Return; we're done
- return False
+ return (False, content)
else:
self.logger.info('Sent Revolt notification.')
@@ -326,9 +350,9 @@ class NotifyRevolt(NotifyBase):
self.logger.warning(
'A Connection error occurred posting to Revolt.')
self.logger.debug('Socket Exception: %s' % str(e))
- return False
+ return (False, content)
- return True
+ return (True, content)
def url(self, privacy=False, *args, **kwargs):
"""
@@ -336,32 +360,37 @@ class NotifyRevolt(NotifyBase):
"""
+ # Define any URL parameters
params = {}
- if self.embed_img:
- params['embed_img'] = self.embed_img
+ if self.icon_url:
+ params['icon_url'] = self.icon_url
- if self.embed_url:
- params['embed_url'] = self.embed_url
+ if self.link:
+ params['url'] = self.link
params.update(self.url_parameters(privacy=privacy, *args, **kwargs))
- return '{schema}://{bot_token}/{channel_id}/?{params}'.format(
+ return '{schema}://{bot_token}/{targets}/?{params}'.format(
schema=self.secure_protocol,
bot_token=self.pprint(self.bot_token, privacy, safe=''),
- channel_id=self.pprint(self.channel_id, privacy, safe=''),
+ targets='/'.join(
+ [self.pprint(x, privacy, safe='') for x in self.targets]),
params=NotifyRevolt.urlencode(params),
)
+ def __len__(self):
+ """
+ Returns the number of targets associated with this notification
+ """
+ return 1 if not self.targets else len(self.targets)
+
@staticmethod
def parse_url(url):
"""
Parses the URL and returns enough arguments that can allow
us to re-instantiate this object.
- Syntax:
- revolt://bot_token/channel_id
-
"""
results = NotifyBase.parse_url(url, verify_host=False)
if not results:
@@ -371,44 +400,37 @@ class NotifyRevolt(NotifyBase):
# Store our bot token
bot_token = NotifyRevolt.unquote(results['host'])
- # Now fetch the channel id
- try:
- channel_id = \
- NotifyRevolt.split_path(results['fullpath'])[0]
-
- except IndexError:
- # Force some bad values that will get caught
- # in parsing later
- channel_id = None
+ # Now fetch the Channel IDs
+ targets = NotifyRevolt.split_path(results['fullpath'])
results['bot_token'] = bot_token
- results['channel_id'] = channel_id
+ results['targets'] = targets
- # Text To Speech
- results['tts'] = parse_bool(results['qsd'].get('tts', False))
+ # Support the 'to' variable so that we can support rooms this way too
+ # The 'to' makes it easier to use yaml configuration
+ if 'to' in results['qsd'] and len(results['qsd']['to']):
+ results['targets'] += \
+ NotifyRevolt.parse_list(results['qsd']['to'])
# Support channel id on the URL string (if specified)
- if 'channel_id' in results['qsd']:
- results['channel_id'] = \
- NotifyRevolt.unquote(results['qsd']['channel_id'])
+ if 'channel' in results['qsd']:
+ results['targets'] += \
+ NotifyRevolt.parse_list(results['qsd']['channel'])
# Support bot token on the URL string (if specified)
if 'bot_token' in results['qsd']:
results['bot_token'] = \
NotifyRevolt.unquote(results['qsd']['bot_token'])
- # Extract avatar url if it was specified
- if 'embed_img' in results['qsd']:
- results['embed_img'] = \
- NotifyRevolt.unquote(results['qsd']['embed_img'])
+ if 'icon_url' in results['qsd']:
+ results['icon_url'] = \
+ NotifyRevolt.unquote(results['qsd']['icon_url'])
- if 'custom_img' in results['qsd']:
- results['custom_img'] = \
- NotifyRevolt.unquote(results['qsd']['custom_img'])
+ if 'url' in results['qsd']:
+ results['link'] = NotifyRevolt.unquote(results['qsd']['url'])
- elif 'embed_url' in results['qsd']:
- results['embed_url'] = \
- NotifyRevolt.unquote(results['qsd']['embed_url'])
+ if 'format' not in results['qsd'] and (
+ 'url' in results or 'icon_url' in results):
# Markdown is implied
results['format'] = NotifyFormat.MARKDOWN
diff --git a/test/test_plugin_revolt.py b/test/test_plugin_revolt.py
index 1701b782..9db9f985 100644
--- a/test/test_plugin_revolt.py
+++ b/test/test_plugin_revolt.py
@@ -29,7 +29,7 @@
import os
from unittest import mock
from datetime import datetime, timedelta
-from datetime import timezone
+from json import dumps
import pytest
import requests
@@ -51,6 +51,15 @@ logging.disable(logging.CRITICAL)
# Attachment Directory
TEST_VAR_DIR = os.path.join(os.path.dirname(__file__), 'var')
+# Prepare a Valid Response
+REVOLT_GOOD_RESPONSE = dumps({
+ '_id': 'AAAPWPMMQA2JJB59BR2EASWWWW',
+ 'nonce': '01HPWPPMDJABC2FTDG54CBKKKS',
+ 'channel': '00000000000000000000000000',
+ 'author': '011244Q9S8NCS67KMM9543W7JJ',
+ 'content': 'test',
+})
+
# Our Testing URLs
apprise_url_tests = (
('revolt://', {
@@ -62,82 +71,128 @@ apprise_url_tests = (
}),
# No channel_id specified
('revolt://%s' % ('i' * 24), {
+ 'instance': NotifyRevolt,
+ # Notify will fail
+ 'response': False,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
+ }),
+ # channel_id specified on url, but no Bot Token
+ ('revolt://?channel=%s' % ('i' * 24), {
'instance': TypeError,
}),
# channel_id specified on url
- ('revolt://?channel_id=%s' % ('i' * 24), {
- 'instance': TypeError,
+ ('revolt://%s/?channel=%s' % ('i' * 24, 'i' * 24), {
+ 'instance': NotifyRevolt,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
+ }),
+ ('revolt://%s/?to=%s' % ('i' * 24, 'i' * 24), {
+ 'instance': NotifyRevolt,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
+ }),
+ ('revolt://%s/?to=%s' % ('i' * 24, 'i' * 24), {
+ 'instance': NotifyRevolt,
+ # an invalid JSON Response
+ 'requests_response_text': '{',
+ }),
+ # channel_id specified on url
+ ('revolt://%s/?channel=%s,%%20' % ('i' * 24, 'i' * 24), {
+ 'instance': NotifyRevolt,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
}),
# Provide both a bot token and a channel id
('revolt://%s/%s' % ('i' * 24, 't' * 64), {
'instance': NotifyRevolt,
- 'requests_response_code': requests.codes.no_content,
+ 'requests_response_code': requests.codes.ok,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
}),
- # Provide a temporary username
- ('revolt://l2g@%s/%s' % ('i' * 24, 't' * 64), {
+ ('revolt://_?bot_token=%s&channel=%s' % ('i' * 24, 't' * 64), {
'instance': NotifyRevolt,
- 'requests_response_code': requests.codes.no_content,
- }),
- ('revolt://l2g@_?bot_token=%s&channel_id=%s' % ('i' * 24, 't' * 64), {
- 'instance': NotifyRevolt,
- 'requests_response_code': requests.codes.no_content,
- }),
- # test custom_img= field
- ('revolt://%s/%s?format=markdown&custom_img=Yes' % (
- 'i' * 24, 't' * 64), {
- 'instance': NotifyRevolt,
- 'requests_response_code': requests.codes.no_content,
- }),
- ('revolt://%s/%s?format=markdown&custom_img=No' % (
- 'i' * 24, 't' * 64), {
- 'instance': NotifyRevolt,
- 'requests_response_code': requests.codes.no_content,
+ 'requests_response_code': requests.codes.ok,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
}),
# different format support
('revolt://%s/%s?format=markdown' % ('i' * 24, 't' * 64), {
'instance': NotifyRevolt,
- 'requests_response_code': requests.codes.no_content,
+ 'requests_response_code': requests.codes.ok,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
}),
('revolt://%s/%s?format=text' % ('i' * 24, 't' * 64), {
'instance': NotifyRevolt,
- 'requests_response_code': requests.codes.no_content,
+ 'requests_response_code': requests.codes.ok,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
}),
- # Test with embed_url (title link)
- ('revolt://%s/%s?hmarkdown=true&embed_url=http://localhost' % (
+ # Test with url
+ ('revolt://%s/%s?url=http://localhost' % (
'i' * 24, 't' * 64), {
'instance': NotifyRevolt,
- 'requests_response_code': requests.codes.no_content,
+ 'requests_response_code': requests.codes.ok,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
}),
- # Test with avatar URL
- ('revolt://%s/%s?embed_img=http://localhost/test.jpg' % (
+ # URL implies markdown unless explicitly set otherwise
+ ('revolt://%s/%s?format=text&url=http://localhost' % (
'i' * 24, 't' * 64), {
'instance': NotifyRevolt,
- 'requests_response_code': requests.codes.no_content,
+ 'requests_response_code': requests.codes.ok,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
}),
- # Test without image set
+ # Test with Icon URL
+ ('revolt://%s/%s?icon_url=http://localhost/test.jpg' % (
+ 'i' * 24, 't' * 64), {
+ 'instance': NotifyRevolt,
+ 'requests_response_code': requests.codes.ok,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
+ }),
+ # Icon URL implies markdown unless explicitly set otherwise
+ ('revolt://%s/%s?format=text&icon_url=http://localhost/test.jpg' % (
+ 'i' * 24, 't' * 64), {
+ 'instance': NotifyRevolt,
+ 'requests_response_code': requests.codes.ok,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
+ }),
+ # Test without any image set
('revolt://%s/%s' % ('i' * 24, 't' * 64), {
'instance': NotifyRevolt,
- 'requests_response_code': requests.codes.no_content,
+ 'requests_response_code': requests.codes.ok,
# don't include an image by default
- 'embed_img': False,
+ 'include_image': False,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
}),
('revolt://%s/%s/' % ('a' * 24, 'b' * 64), {
'instance': NotifyRevolt,
# force a failure
'response': False,
'requests_response_code': requests.codes.internal_server_error,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
}),
('revolt://%s/%s/' % ('a' * 24, 'b' * 64), {
'instance': NotifyRevolt,
# throw a bizzare code forcing us to fail to look it up
'response': False,
'requests_response_code': 999,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
}),
('revolt://%s/%s/' % ('a' * 24, 'b' * 64), {
'instance': NotifyRevolt,
# Throws a series of connection and transfer exceptions when this flag
# is set and tests that we gracfully handle them
'test_requests_exceptions': True,
+ # Our response expected server response
+ 'requests_response_text': REVOLT_GOOD_RESPONSE,
}),
)
@@ -155,7 +210,7 @@ def test_plugin_revolt_urls():
@mock.patch('requests.post')
def test_plugin_revolt_notifications(mock_post):
"""
- NotifyRevolt() Notifications/Ping Support
+ NotifyRevolt() Notifications
"""
@@ -166,6 +221,7 @@ def test_plugin_revolt_notifications(mock_post):
# Prepare Mock
mock_post.return_value = requests.Request()
mock_post.return_value.status_code = requests.codes.ok
+ mock_post.return_value.content = REVOLT_GOOD_RESPONSE
# Test our header parsing when not lead with a header
body = """
@@ -179,7 +235,7 @@ def test_plugin_revolt_notifications(mock_post):
assert isinstance(results, dict)
assert results['user'] is None
assert results['bot_token'] == bot_token
- assert results['channel_id'] == channel_id
+ assert results['targets'] == [channel_id, ]
assert results['password'] is None
assert results['port'] is None
assert results['host'] == bot_token
@@ -205,7 +261,7 @@ def test_plugin_revolt_notifications(mock_post):
assert isinstance(results, dict)
assert results['user'] is None
assert results['bot_token'] == bot_token
- assert results['channel_id'] == channel_id
+ assert results['targets'] == [channel_id, ]
assert results['password'] is None
assert results['port'] is None
assert results['host'] == bot_token
@@ -224,48 +280,42 @@ def test_plugin_revolt_notifications(mock_post):
@mock.patch('requests.post')
-def test_plugin_revolt_general(mock_post):
+@mock.patch('time.sleep')
+def test_plugin_revolt_general(mock_sleep, mock_post):
"""
NotifyRevolt() General Checks
"""
+ # Prevent throttling
+ mock_sleep.return_value = True
+
# Turn off clock skew for local testing
NotifyRevolt.clock_skew = timedelta(seconds=0)
- # Epoch time:
- epoch = datetime.fromtimestamp(0, timezone.utc)
# Initialize some generic (but valid) tokens
bot_token = 'A' * 24
- channel_id = 'B' * 64
+ channel_id = ','.join(['B' * 32, 'C' * 32]) + ', ,%%'
# Prepare Mock
mock_post.return_value = requests.Request()
mock_post.return_value.status_code = requests.codes.ok
- mock_post.return_value.content = ''
+ mock_post.return_value.content = REVOLT_GOOD_RESPONSE
mock_post.return_value.headers = {
- 'X-RateLimit-Reset': (
- datetime.now(timezone.utc) - epoch).total_seconds(),
- 'X-RateLimit-Remaining': 1,
+ 'X-RateLimit-Remaining': 0,
+ 'X-RateLimit-Reset-After': 1,
}
# Invalid bot_token
with pytest.raises(TypeError):
- NotifyRevolt(bot_token=None, channel_id=channel_id)
+ NotifyRevolt(bot_token=None, targets=channel_id)
# Invalid bot_token (whitespace)
with pytest.raises(TypeError):
- NotifyRevolt(bot_token=" ", channel_id=channel_id)
-
- # Invalid channel_id
- with pytest.raises(TypeError):
- NotifyRevolt(bot_token=bot_token, channel_id=None)
- # Invalid channel_id (whitespace)
- with pytest.raises(TypeError):
- NotifyRevolt(bot_token=bot_token, channel_id=" ")
+ NotifyRevolt(bot_token=" ", targets=channel_id)
obj = NotifyRevolt(
bot_token=bot_token,
- channel_id=channel_id)
+ targets=channel_id)
assert obj.ratelimit_remaining == 1
# Test that we get a string response
@@ -277,9 +327,8 @@ def test_plugin_revolt_general(mock_post):
# Force a case where there are no more remaining posts allowed
mock_post.return_value.headers = {
- 'X-RateLimit-Reset': (
- datetime.now(timezone.utc) - epoch).total_seconds(),
'X-RateLimit-Remaining': 0,
+ 'X-RateLimit-Reset-After': 0,
}
# This call includes an image with it's payload:
@@ -289,31 +338,31 @@ def test_plugin_revolt_general(mock_post):
# behind the scenes, it should cause us to update our rate limit
assert obj.send(body="test") is True
assert obj.ratelimit_remaining == 0
+ assert isinstance(obj.ratelimit_reset, datetime)
# This should cause us to block
mock_post.return_value.headers = {
- 'X-RateLimit-Reset': (
- datetime.now(timezone.utc) - epoch).total_seconds(),
- 'X-RateLimit-Remaining': 10,
+ 'X-RateLimit-Remaining': 0,
+ 'X-RateLimit-Reset-After': 3000,
}
assert obj.send(body="test") is True
- assert obj.ratelimit_remaining == 10
+ assert obj.ratelimit_remaining == 0
+ assert isinstance(obj.ratelimit_reset, datetime)
# Reset our variable back to 1
mock_post.return_value.headers = {
- 'X-RateLimit-Reset': (
- datetime.now(timezone.utc) - epoch).total_seconds(),
- 'X-RateLimit-Remaining': 1,
+ 'X-RateLimit-Remaining': 0,
+ 'X-RateLimit-Reset-After': 10000,
}
- # Handle cases where our epoch time is wrong
- del mock_post.return_value.headers['X-RateLimit-Reset']
+ del mock_post.return_value.headers['X-RateLimit-Remaining']
assert obj.send(body="test") is True
+ assert obj.ratelimit_remaining == 0
+ assert isinstance(obj.ratelimit_reset, datetime)
# Return our object, but place it in the future forcing us to block
mock_post.return_value.headers = {
- 'X-RateLimit-Reset': (
- datetime.now(timezone.utc) - epoch).total_seconds() + 1,
'X-RateLimit-Remaining': 0,
+ 'X-RateLimit-Reset-After': 0,
}
obj.ratelimit_remaining = 0
@@ -329,9 +378,8 @@ def test_plugin_revolt_general(mock_post):
# Return our object, but place it in the future forcing us to block
mock_post.return_value.status_code = requests.codes.ok
mock_post.return_value.headers = {
- 'X-RateLimit-Reset': (
- datetime.now(timezone.utc) - epoch).total_seconds() - 1,
'X-RateLimit-Remaining': 0,
+ 'X-RateLimit-Reset-After': 0,
}
assert obj.send(body="test") is True
@@ -340,9 +388,8 @@ def test_plugin_revolt_general(mock_post):
# Return our headers to normal
mock_post.return_value.headers = {
- 'X-RateLimit-Reset': (
- datetime.now(timezone.utc) - epoch).total_seconds(),
- 'X-RateLimit-Remaining': 1,
+ 'X-RateLimit-Remaining': 0,
+ 'X-RateLimit-Reset-After': 1,
}
# This call includes an image with it's payload:
@@ -380,6 +427,7 @@ def test_plugin_revolt_overflow(mock_post):
# Prepare Mock
mock_post.return_value = requests.Request()
mock_post.return_value.status_code = requests.codes.ok
+ mock_post.return_value.content = REVOLT_GOOD_RESPONSE
# Some variables we use to control the data we work with
body_len = 2005
@@ -401,7 +449,7 @@ def test_plugin_revolt_overflow(mock_post):
assert isinstance(results, dict)
assert results['user'] is None
assert results['bot_token'] == bot_token
- assert results['channel_id'] == channel_id
+ assert results['targets'] == [channel_id, ]
assert results['password'] is None
assert results['port'] is None
assert results['host'] == bot_token
@@ -436,6 +484,7 @@ def test_plugin_revolt_markdown_extra(mock_post):
# Prepare Mock
mock_post.return_value = requests.Request()
mock_post.return_value.status_code = requests.codes.ok
+ mock_post.return_value.content = REVOLT_GOOD_RESPONSE
# Reset our apprise object
a = Apprise()