2017-12-25 21:07:41 +01:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
2019-02-01 03:05:56 +01:00
|
|
|
# Copyright (C) 2019 Chris Caron <lead2gold@gmail.com>
|
|
|
|
# All rights reserved.
|
2017-12-25 21:07:41 +01:00
|
|
|
#
|
2019-02-01 03:05:56 +01:00
|
|
|
# This code is licensed under the MIT License.
|
2017-12-25 21:07:41 +01:00
|
|
|
#
|
2019-02-01 03:05:56 +01:00
|
|
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
# of this software and associated documentation files(the "Software"), to deal
|
|
|
|
# in the Software without restriction, including without limitation the rights
|
|
|
|
# to use, copy, modify, merge, publish, distribute, sublicense, and / or sell
|
|
|
|
# copies of the Software, and to permit persons to whom the Software is
|
|
|
|
# furnished to do so, subject to the following conditions :
|
2017-12-25 21:07:41 +01:00
|
|
|
#
|
2019-02-01 03:05:56 +01:00
|
|
|
# The above copyright notice and this permission notice shall be included in
|
|
|
|
# all copies or substantial portions of the Software.
|
2017-12-25 21:07:41 +01:00
|
|
|
#
|
2019-02-01 03:05:56 +01:00
|
|
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.IN NO EVENT SHALL THE
|
|
|
|
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
# THE SOFTWARE.
|
2017-12-25 21:07:41 +01:00
|
|
|
|
2019-03-27 04:06:20 +01:00
|
|
|
import six
|
|
|
|
import mock
|
2019-06-30 00:27:59 +02:00
|
|
|
import requests
|
|
|
|
from json import dumps
|
|
|
|
from datetime import datetime
|
2017-12-25 21:07:41 +01:00
|
|
|
from apprise import plugins
|
|
|
|
|
2019-02-25 07:02:29 +01:00
|
|
|
# Disable logging for a cleaner testing output
|
|
|
|
import logging
|
|
|
|
logging.disable(logging.CRITICAL)
|
|
|
|
|
2017-12-25 21:07:41 +01:00
|
|
|
|
2019-06-30 00:27:59 +02:00
|
|
|
def test_twitter_plugin_init():
|
2017-12-25 21:07:41 +01:00
|
|
|
"""
|
|
|
|
API: NotifyTwitter Plugin() (pt2)
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
try:
|
|
|
|
plugins.NotifyTwitter(
|
|
|
|
ckey=None, csecret=None, akey=None, asecret=None)
|
|
|
|
assert False
|
|
|
|
except TypeError:
|
|
|
|
# All keys set to none
|
|
|
|
assert True
|
|
|
|
|
|
|
|
try:
|
|
|
|
plugins.NotifyTwitter(
|
|
|
|
ckey='value', csecret=None, akey=None, asecret=None)
|
|
|
|
assert False
|
|
|
|
except TypeError:
|
|
|
|
# csecret not set
|
|
|
|
assert True
|
|
|
|
|
|
|
|
try:
|
|
|
|
plugins.NotifyTwitter(
|
|
|
|
ckey='value', csecret='value', akey=None, asecret=None)
|
|
|
|
assert False
|
|
|
|
except TypeError:
|
|
|
|
# akey not set
|
|
|
|
assert True
|
|
|
|
|
|
|
|
try:
|
|
|
|
plugins.NotifyTwitter(
|
|
|
|
ckey='value', csecret='value', akey='value', asecret=None)
|
|
|
|
assert False
|
|
|
|
except TypeError:
|
|
|
|
# asecret not set
|
|
|
|
assert True
|
|
|
|
|
|
|
|
try:
|
|
|
|
plugins.NotifyTwitter(
|
|
|
|
ckey='value', csecret='value', akey='value', asecret='value')
|
|
|
|
assert True
|
2019-06-30 00:27:59 +02:00
|
|
|
except TypeError:
|
|
|
|
# user not set; but this is okay
|
|
|
|
# We should not reach here
|
|
|
|
assert False
|
2017-12-25 21:07:41 +01:00
|
|
|
|
|
|
|
try:
|
2019-06-30 00:27:59 +02:00
|
|
|
plugins.NotifyTwitter(
|
2017-12-25 21:07:41 +01:00
|
|
|
ckey='value', csecret='value', akey='value', asecret='value',
|
2019-06-30 00:27:59 +02:00
|
|
|
user='l2gnux')
|
2017-12-25 21:07:41 +01:00
|
|
|
# We should initialize properly
|
|
|
|
assert True
|
|
|
|
|
|
|
|
except TypeError:
|
|
|
|
# We should not reach here
|
|
|
|
assert False
|
|
|
|
|
|
|
|
|
2019-06-30 00:27:59 +02:00
|
|
|
@mock.patch('requests.get')
|
|
|
|
@mock.patch('requests.post')
|
|
|
|
def test_notify_twitter_plugin_general(mock_post, mock_get):
|
|
|
|
"""
|
|
|
|
API: NotifyTwitter() General Tests
|
2017-12-25 21:07:41 +01:00
|
|
|
|
2019-06-30 00:27:59 +02:00
|
|
|
"""
|
|
|
|
ckey = 'ckey'
|
|
|
|
csecret = 'csecret'
|
|
|
|
akey = 'akey'
|
|
|
|
asecret = 'asecret'
|
|
|
|
screen_name = 'apprise'
|
|
|
|
|
|
|
|
response_obj = [{
|
|
|
|
'screen_name': screen_name,
|
|
|
|
'id': 9876,
|
|
|
|
}]
|
|
|
|
|
|
|
|
# Disable Throttling to speed testing
|
|
|
|
plugins.NotifyBase.request_rate_per_sec = 0
|
2017-12-25 21:07:41 +01:00
|
|
|
|
2019-06-30 00:27:59 +02:00
|
|
|
# Epoch time:
|
|
|
|
epoch = datetime.utcfromtimestamp(0)
|
|
|
|
|
|
|
|
request = mock.Mock()
|
|
|
|
request.content = dumps(response_obj)
|
|
|
|
request.status_code = requests.codes.ok
|
|
|
|
request.headers = {
|
|
|
|
'x-rate-limit-reset': (datetime.utcnow() - epoch).total_seconds(),
|
|
|
|
'x-rate-limit-remaining': 1,
|
|
|
|
}
|
|
|
|
|
|
|
|
# Prepare Mock
|
|
|
|
mock_get.return_value = request
|
|
|
|
mock_post.return_value = request
|
|
|
|
|
|
|
|
# Variation Initializations
|
|
|
|
obj = plugins.NotifyTwitter(
|
|
|
|
ckey=ckey,
|
|
|
|
csecret=csecret,
|
|
|
|
akey=akey,
|
|
|
|
asecret=asecret,
|
|
|
|
targets=screen_name)
|
|
|
|
|
|
|
|
assert isinstance(obj, plugins.NotifyTwitter) is True
|
|
|
|
assert isinstance(obj.url(), six.string_types) is True
|
|
|
|
|
|
|
|
# apprise room was found
|
|
|
|
assert obj.send(body="test") is True
|
|
|
|
|
|
|
|
# Change our status code and try again
|
|
|
|
request.status_code = 403
|
|
|
|
assert obj.send(body="test") is False
|
|
|
|
assert obj.ratelimit_remaining == 1
|
|
|
|
|
|
|
|
# Return the status
|
|
|
|
request.status_code = requests.codes.ok
|
|
|
|
# Force a reset
|
|
|
|
request.headers['x-rate-limit-remaining'] = 0
|
|
|
|
# behind the scenes, it should cause us to update our rate limit
|
|
|
|
assert obj.send(body="test") is True
|
|
|
|
assert obj.ratelimit_remaining == 0
|
|
|
|
|
|
|
|
# This should cause us to block
|
|
|
|
request.headers['x-rate-limit-remaining'] = 10
|
|
|
|
assert obj.send(body="test") is True
|
|
|
|
assert obj.ratelimit_remaining == 10
|
|
|
|
|
|
|
|
# Handle cases where we simply couldn't get this field
|
|
|
|
del request.headers['x-rate-limit-remaining']
|
|
|
|
assert obj.send(body="test") is True
|
|
|
|
# It remains set to the last value
|
|
|
|
assert obj.ratelimit_remaining == 10
|
|
|
|
|
|
|
|
# Reset our variable back to 1
|
|
|
|
request.headers['x-rate-limit-remaining'] = 1
|
|
|
|
|
|
|
|
# Handle cases where our epoch time is wrong
|
|
|
|
del request.headers['x-rate-limit-reset']
|
|
|
|
assert obj.send(body="test") is True
|
|
|
|
|
|
|
|
# Return our object, but place it in the future forcing us to block
|
|
|
|
request.headers['x-rate-limit-reset'] = \
|
|
|
|
(datetime.utcnow() - epoch).total_seconds() + 1
|
|
|
|
request.headers['x-rate-limit-remaining'] = 0
|
|
|
|
obj.ratelimit_remaining = 0
|
|
|
|
assert obj.send(body="test") is True
|
|
|
|
|
|
|
|
# Return our object, but place it in the future forcing us to block
|
|
|
|
request.headers['x-rate-limit-reset'] = \
|
|
|
|
(datetime.utcnow() - epoch).total_seconds() - 1
|
|
|
|
request.headers['x-rate-limit-remaining'] = 0
|
|
|
|
obj.ratelimit_remaining = 0
|
|
|
|
assert obj.send(body="test") is True
|
|
|
|
|
|
|
|
# Return our limits to always work
|
|
|
|
request.headers['x-rate-limit-reset'] = \
|
|
|
|
(datetime.utcnow() - epoch).total_seconds()
|
|
|
|
request.headers['x-rate-limit-remaining'] = 1
|
|
|
|
obj.ratelimit_remaining = 1
|
|
|
|
|
|
|
|
# Alter pending targets
|
|
|
|
obj.targets.append('usera')
|
|
|
|
request.content = dumps(response_obj)
|
|
|
|
response_obj = [{
|
|
|
|
'screen_name': 'usera',
|
|
|
|
'id': 1234,
|
|
|
|
}]
|
|
|
|
|
|
|
|
assert obj.send(body="test") is True
|
|
|
|
|
|
|
|
# Flush our cache forcing it's re-creating
|
|
|
|
del plugins.NotifyTwitter._user_cache
|
|
|
|
assert obj.send(body="test") is True
|
|
|
|
|
|
|
|
# Cause content response to be None
|
|
|
|
request.content = None
|
|
|
|
assert obj.send(body="test") is True
|
|
|
|
|
|
|
|
# Invalid JSON
|
|
|
|
request.content = '{'
|
|
|
|
assert obj.send(body="test") is True
|
|
|
|
|
|
|
|
# Return it to a parseable string
|
|
|
|
request.content = '{}'
|
|
|
|
|
|
|
|
results = plugins.NotifyTwitter.parse_url(
|
|
|
|
'twitter://{}/{}/{}/{}?to={}'.format(
|
|
|
|
ckey, csecret, akey, asecret, screen_name))
|
|
|
|
assert isinstance(results, dict) is True
|
|
|
|
assert screen_name in results['targets']
|
|
|
|
|
|
|
|
# cause a json parsing issue now
|
|
|
|
response_obj = None
|
|
|
|
assert obj.send(body="test") is True
|
|
|
|
|
|
|
|
response_obj = '{'
|
|
|
|
assert obj.send(body="test") is True
|
|
|
|
|
|
|
|
# Set ourselves up to handle whoami calls
|
|
|
|
|
|
|
|
# Flush out our cache
|
|
|
|
del plugins.NotifyTwitter._user_cache
|
|
|
|
|
|
|
|
response_obj = {
|
|
|
|
'screen_name': screen_name,
|
|
|
|
'id': 9876,
|
|
|
|
}
|
|
|
|
request.content = dumps(response_obj)
|
|
|
|
|
|
|
|
obj = plugins.NotifyTwitter(
|
|
|
|
ckey=ckey,
|
|
|
|
csecret=csecret,
|
|
|
|
akey=akey,
|
|
|
|
asecret=asecret)
|
|
|
|
|
|
|
|
assert obj.send(body="test") is True
|
|
|
|
|
|
|
|
# Alter the key forcing us to look up a new value of ourselves again
|
|
|
|
del plugins.NotifyTwitter._user_cache
|
|
|
|
del plugins.NotifyTwitter._whoami_cache
|
|
|
|
obj.ckey = 'different.then.it.was'
|
|
|
|
assert obj.send(body="test") is True
|
|
|
|
|
|
|
|
del plugins.NotifyTwitter._whoami_cache
|
|
|
|
obj.ckey = 'different.again'
|
|
|
|
assert obj.send(body="test") is True
|