mirror of
https://github.com/caronc/apprise.git
synced 2024-11-29 19:43:30 +01:00
c81d2465e4
Instead of needing to individually disable throttling on a per-plugin basis, this fixture takes care of all notifiers in `NOTIFY_MODULE_MAP` automatically. With `autouse=True`, there is no need to activate it manually.
493 lines
16 KiB
Python
493 lines
16 KiB
Python
# -*- coding: utf-8 -*-
|
|
#
|
|
# Copyright (C) 2022 Chris Caron <lead2gold@gmail.com>
|
|
# All rights reserved.
|
|
#
|
|
# This code is licensed under the MIT License.
|
|
#
|
|
# 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 :
|
|
#
|
|
# The above copyright notice and this permission notice shall be included in
|
|
# all copies or substantial portions of the Software.
|
|
#
|
|
# 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.
|
|
import os
|
|
import json
|
|
from unittest import mock
|
|
|
|
import requests
|
|
import apprise
|
|
from helpers import AppriseURLTester
|
|
|
|
from apprise.plugins.NotifyNtfy import NtfyPriority, NotifyNtfy
|
|
|
|
# Disable logging for a cleaner testing output
|
|
import logging
|
|
logging.disable(logging.CRITICAL)
|
|
|
|
# Attachment Directory
|
|
TEST_VAR_DIR = os.path.join(os.path.dirname(__file__), 'var')
|
|
|
|
# For testing our return response
|
|
GOOD_RESPONSE_TEXT = {
|
|
'code': '0',
|
|
'error': 'success',
|
|
}
|
|
|
|
# Our Testing URLs
|
|
apprise_url_tests = (
|
|
('ntfy://', {
|
|
# Initializes okay (as cloud mode) but has no topics to notify
|
|
'instance': NotifyNtfy,
|
|
# invalid topics specified (nothing to notify)
|
|
# as a result the response type will be false
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
'response': False,
|
|
}),
|
|
('ntfys://', {
|
|
# Initializes okay (as cloud mode) but has no topics to notify
|
|
'instance': NotifyNtfy,
|
|
# invalid topics specified (nothing to notify)
|
|
# as a result the response type will be false
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
'response': False,
|
|
}),
|
|
('ntfy://:@/', {
|
|
# Initializes okay (as cloud mode) but has no topics to notify
|
|
'instance': NotifyNtfy,
|
|
# invalid topics specified (nothing to notify)
|
|
# as a result the response type will be false
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
'response': False,
|
|
}),
|
|
# No topics
|
|
('ntfy://user:pass@localhost?mode=private', {
|
|
'instance': NotifyNtfy,
|
|
# invalid topics specified (nothing to notify)
|
|
# as a result the response type will be false
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
'response': False,
|
|
}),
|
|
# No valid topics
|
|
('ntfy://user:pass@localhost/#/!/@', {
|
|
'instance': NotifyNtfy,
|
|
# invalid topics specified (nothing to notify)
|
|
# as a result the response type will be false
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
'response': False,
|
|
}),
|
|
# user/pass combos
|
|
('ntfy://user@localhost/topic/', {
|
|
'instance': NotifyNtfy,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# Ntfy cloud mode (enforced)
|
|
('ntfy://ntfy.sh/topic1/topic2/', {
|
|
'instance': NotifyNtfy,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# No user/pass combo
|
|
('ntfy://localhost/topic1/topic2/', {
|
|
'instance': NotifyNtfy,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# A Email Testing
|
|
('ntfy://localhost/topic1/?email=user@gmail.com', {
|
|
'instance': NotifyNtfy,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# Tags
|
|
('ntfy://localhost/topic1/?tags=tag1,tag2,tag3', {
|
|
'instance': NotifyNtfy,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# Delay
|
|
('ntfy://localhost/topic1/?delay=3600', {
|
|
'instance': NotifyNtfy,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# Title
|
|
('ntfy://localhost/topic1/?title=A%20Great%20Title', {
|
|
'instance': NotifyNtfy,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# Click
|
|
('ntfy://localhost/topic1/?click=yes', {
|
|
'instance': NotifyNtfy,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# Email
|
|
('ntfy://localhost/topic1/?email=user@example.com', {
|
|
'instance': NotifyNtfy,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# Attach
|
|
('ntfy://localhost/topic1/?attach=http://example.com/file.jpg', {
|
|
'instance': NotifyNtfy,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# Attach with filename over-ride
|
|
('ntfy://localhost/topic1/'
|
|
'?attach=http://example.com/file.jpg&filename=smoke.jpg', {
|
|
'instance': NotifyNtfy,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT}),
|
|
# Attach with bad url
|
|
('ntfy://localhost/topic1/?attach=http://-%20', {
|
|
'instance': NotifyNtfy,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# Priority
|
|
('ntfy://localhost/topic1/?priority=default', {
|
|
'instance': NotifyNtfy,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# Priority higher
|
|
('ntfy://localhost/topic1/?priority=high', {
|
|
'instance': NotifyNtfy,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# A topic and port identifier
|
|
('ntfy://user:pass@localhost:8080/topic/', {
|
|
'instance': NotifyNtfy,
|
|
# The response text is expected to be the following on a success
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# A topic (using the to=)
|
|
('ntfys://user:pass@localhost?to=topic', {
|
|
'instance': NotifyNtfy,
|
|
# The response text is expected to be the following on a success
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
('https://just/a/random/host/that/means/nothing', {
|
|
# Nothing transpires from this
|
|
'instance': None
|
|
}),
|
|
# reference the ntfy.sh url
|
|
('https://ntfy.sh?to=topic', {
|
|
'instance': NotifyNtfy,
|
|
# The response text is expected to be the following on a success
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# Several topics
|
|
('ntfy://user:pass@topic1/topic2/topic3/?mode=cloud', {
|
|
'instance': NotifyNtfy,
|
|
# The response text is expected to be the following on a success
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
# Several topics (but do not add ntfy.sh)
|
|
('ntfy://user:pass@ntfy.sh/topic1/topic2/?mode=cloud', {
|
|
'instance': NotifyNtfy,
|
|
# The response text is expected to be the following on a success
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
('ntfys://user:web/token@localhost/topic/?mode=invalid', {
|
|
# Invalid mode
|
|
'instance': TypeError,
|
|
}),
|
|
# Invalid hostname on localhost/private mode
|
|
('ntfys://user:web@-_/topic1/topic2/?mode=private', {
|
|
'instance': None,
|
|
}),
|
|
('ntfy://user:pass@localhost:8089/topic/topic2', {
|
|
'instance': NotifyNtfy,
|
|
# force a failure using basic mode
|
|
'response': False,
|
|
'requests_response_code': requests.codes.internal_server_error,
|
|
}),
|
|
('ntfy://user:pass@localhost:8082/topic', {
|
|
'instance': NotifyNtfy,
|
|
# throw a bizzare code forcing us to fail to look it up
|
|
'response': False,
|
|
'requests_response_code': 999,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
('ntfy://user:pass@localhost:8083/topic1/topic2/', {
|
|
'instance': NotifyNtfy,
|
|
# Throws a series of connection and transfer exceptions when this flag
|
|
# is set and tests that we gracfully handle them
|
|
'test_requests_exceptions': True,
|
|
'requests_response_text': GOOD_RESPONSE_TEXT,
|
|
}),
|
|
)
|
|
|
|
|
|
def test_plugin_ntfy_chat_urls():
|
|
"""
|
|
NotifyNtfy() Apprise URLs
|
|
|
|
"""
|
|
|
|
# Run our general tests
|
|
AppriseURLTester(tests=apprise_url_tests).run_all()
|
|
|
|
|
|
@mock.patch('requests.post')
|
|
def test_plugin_ntfy_attachments(mock_post):
|
|
"""
|
|
NotifyNtfy() Attachment Checks
|
|
|
|
"""
|
|
|
|
# Prepare Mock return object
|
|
response = mock.Mock()
|
|
response.content = GOOD_RESPONSE_TEXT
|
|
response.status_code = requests.codes.ok
|
|
mock_post.return_value = response
|
|
|
|
# Test how the notifications work without attachments as they use the
|
|
# JSON type posting instead
|
|
|
|
# Reset our mock object
|
|
mock_post.reset_mock()
|
|
|
|
# Prepare our object
|
|
obj = apprise.Apprise.instantiate(
|
|
'ntfy://user:pass@localhost:8080/topic')
|
|
|
|
# Send a good attachment
|
|
assert obj.notify(title="hello", body="world")
|
|
assert mock_post.call_count == 1
|
|
|
|
assert mock_post.call_args_list[0][0][0] == \
|
|
'http://localhost:8080'
|
|
|
|
response = json.loads(mock_post.call_args_list[0][1]['data'])
|
|
assert response['topic'] == 'topic'
|
|
assert response['title'] == 'hello'
|
|
assert response['message'] == 'world'
|
|
assert 'attach' not in response
|
|
|
|
# Reset our mock object
|
|
mock_post.reset_mock()
|
|
|
|
# prepare our attachment
|
|
attach = apprise.AppriseAttachment(
|
|
os.path.join(TEST_VAR_DIR, 'apprise-test.gif'))
|
|
|
|
# Prepare our object
|
|
obj = apprise.Apprise.instantiate(
|
|
'ntfy://user:pass@localhost:8084/topic')
|
|
|
|
# Send a good attachment
|
|
assert obj.notify(body="test", attach=attach) is True
|
|
|
|
# Test our call count; includes both image and message
|
|
assert mock_post.call_count == 1
|
|
|
|
assert mock_post.call_args_list[0][0][0] == \
|
|
'http://localhost:8084/topic'
|
|
|
|
assert mock_post.call_args_list[0][1]['params']['message'] == 'test'
|
|
assert 'title' not in mock_post.call_args_list[0][1]['params']
|
|
assert mock_post.call_args_list[0][1]['params']['filename'] == \
|
|
'apprise-test.gif'
|
|
|
|
# Reset our mock object
|
|
mock_post.reset_mock()
|
|
|
|
# Add another attachment so we drop into the area of the PushBullet code
|
|
# that sends remaining attachments (if more detected)
|
|
attach.add(os.path.join(TEST_VAR_DIR, 'apprise-test.png'))
|
|
|
|
# Send our attachments
|
|
assert obj.notify(body="test", title="wonderful", attach=attach) is True
|
|
|
|
# Test our call count
|
|
assert mock_post.call_count == 2
|
|
# Image + Message sent
|
|
assert mock_post.call_args_list[0][0][0] == \
|
|
'http://localhost:8084/topic'
|
|
assert mock_post.call_args_list[0][1]['params']['message'] == \
|
|
'test'
|
|
assert mock_post.call_args_list[0][1]['params']['title'] == \
|
|
'wonderful'
|
|
assert mock_post.call_args_list[0][1]['params']['filename'] == \
|
|
'apprise-test.gif'
|
|
|
|
# Image no 2 (no message)
|
|
assert mock_post.call_args_list[1][0][0] == \
|
|
'http://localhost:8084/topic'
|
|
assert 'message' not in mock_post.call_args_list[1][1]['params']
|
|
assert 'title' not in mock_post.call_args_list[1][1]['params']
|
|
assert mock_post.call_args_list[1][1]['params']['filename'] == \
|
|
'apprise-test.png'
|
|
|
|
# Reset our mock object
|
|
mock_post.reset_mock()
|
|
|
|
# An invalid attachment will cause a failure
|
|
path = os.path.join(TEST_VAR_DIR, '/invalid/path/to/an/invalid/file.jpg')
|
|
attach = apprise.AppriseAttachment(path)
|
|
assert obj.notify(body="test", attach=attach) is False
|
|
|
|
# Test our call count
|
|
assert mock_post.call_count == 0
|
|
|
|
# prepare our attachment
|
|
attach = apprise.AppriseAttachment(
|
|
os.path.join(TEST_VAR_DIR, 'apprise-test.gif'))
|
|
|
|
# Throw an exception on the first call to requests.post()
|
|
mock_post.return_value = None
|
|
for side_effect in (requests.RequestException(), OSError()):
|
|
mock_post.side_effect = side_effect
|
|
|
|
# We'll fail now because of our error handling
|
|
assert obj.send(body="test", attach=attach) is False
|
|
|
|
|
|
@mock.patch('requests.post')
|
|
def test_plugin_custom_ntfy_edge_cases(mock_post):
|
|
"""
|
|
NotifyNtfy() Edge Cases
|
|
|
|
"""
|
|
|
|
# Prepare our response
|
|
response = requests.Request()
|
|
response.status_code = requests.codes.ok
|
|
response.content = json.dumps(GOOD_RESPONSE_TEXT)
|
|
|
|
# Prepare Mock
|
|
mock_post.return_value = response
|
|
|
|
results = NotifyNtfy.parse_url(
|
|
'ntfys://abc---,topic2,~~,,?priority=max&tags=smile,de')
|
|
|
|
assert isinstance(results, dict)
|
|
assert results['user'] is None
|
|
assert results['password'] is None
|
|
assert results['port'] is None
|
|
assert results['host'] == 'abc---,topic2,~~,,'
|
|
assert results['fullpath'] is None
|
|
assert results['path'] is None
|
|
assert results['query'] is None
|
|
assert results['schema'] == 'ntfys'
|
|
assert results['url'] == 'ntfys://abc---,topic2,~~,,'
|
|
assert isinstance(results['qsd:'], dict) is True
|
|
assert results['qsd']['priority'] == 'max'
|
|
assert results['qsd']['tags'] == 'smile,de'
|
|
|
|
instance = NotifyNtfy(**results)
|
|
assert isinstance(instance, NotifyNtfy)
|
|
assert len(instance.topics) == 2
|
|
assert 'abc---' in instance.topics
|
|
assert 'topic2' in instance.topics
|
|
|
|
results = NotifyNtfy.parse_url(
|
|
'ntfy://localhost/topic1/'
|
|
'?attach=http://example.com/file.jpg&filename=smoke.jpg')
|
|
|
|
assert isinstance(results, dict)
|
|
assert results['user'] is None
|
|
assert results['password'] is None
|
|
assert results['port'] is None
|
|
assert results['host'] == 'localhost'
|
|
assert results['fullpath'] == '/topic1/'
|
|
assert results['path'] == '/topic1/'
|
|
assert results['query'] is None
|
|
assert results['schema'] == 'ntfy'
|
|
assert results['url'] == 'ntfy://localhost/topic1/'
|
|
assert results['attach'] == 'http://example.com/file.jpg'
|
|
assert results['filename'] == 'smoke.jpg'
|
|
|
|
instance = NotifyNtfy(**results)
|
|
assert isinstance(instance, NotifyNtfy)
|
|
assert len(instance.topics) == 1
|
|
assert 'topic1' in instance.topics
|
|
|
|
assert instance.notify(
|
|
body='body', title='title',
|
|
notify_type=apprise.NotifyType.INFO) is True
|
|
|
|
# Test our call count
|
|
assert mock_post.call_count == 1
|
|
assert mock_post.call_args_list[0][0][0] == 'http://localhost'
|
|
|
|
response = json.loads(mock_post.call_args_list[0][1]['data'])
|
|
assert response['topic'] == 'topic1'
|
|
assert response['message'] == 'body'
|
|
assert response['title'] == 'title'
|
|
assert response['attach'] == 'http://example.com/file.jpg'
|
|
assert response['filename'] == 'smoke.jpg'
|
|
|
|
|
|
@mock.patch('requests.post')
|
|
@mock.patch('requests.get')
|
|
def test_plugin_ntfy_config_files(mock_post, mock_get):
|
|
"""
|
|
NotifyNtfy() Config File Cases
|
|
"""
|
|
content = """
|
|
urls:
|
|
- ntfy://localhost/topic1:
|
|
- priority: 1
|
|
tag: ntfy_int min
|
|
- priority: "1"
|
|
tag: ntfy_str_int min
|
|
- priority: min
|
|
tag: ntfy_str min
|
|
|
|
# This will take on normal (default) priority
|
|
- priority: invalid
|
|
tag: ntfy_invalid
|
|
|
|
- ntfy://localhost/topic2:
|
|
- priority: 5
|
|
tag: ntfy_int max
|
|
- priority: "5"
|
|
tag: ntfy_str_int max
|
|
- priority: emergency
|
|
tag: ntfy_str max
|
|
- priority: max
|
|
tag: ntfy_str max
|
|
"""
|
|
|
|
# Prepare Mock
|
|
mock_post.return_value = requests.Request()
|
|
mock_post.return_value.status_code = requests.codes.ok
|
|
mock_get.return_value = requests.Request()
|
|
mock_get.return_value.status_code = requests.codes.ok
|
|
|
|
# Create ourselves a config object
|
|
ac = apprise.AppriseConfig()
|
|
assert ac.add_config(content=content) is True
|
|
|
|
aobj = apprise.Apprise()
|
|
|
|
# Add our configuration
|
|
aobj.add(ac)
|
|
|
|
# We should be able to read our 8 servers from that
|
|
# 3x min
|
|
# 4x max
|
|
# 1x invalid (so takes on normal priority)
|
|
assert len(ac.servers()) == 8
|
|
assert len(aobj) == 8
|
|
assert len([x for x in aobj.find(tag='min')]) == 3
|
|
for s in aobj.find(tag='min'):
|
|
assert s.priority == NtfyPriority.MIN
|
|
|
|
assert len([x for x in aobj.find(tag='max')]) == 4
|
|
for s in aobj.find(tag='max'):
|
|
assert s.priority == NtfyPriority.MAX
|
|
|
|
assert len([x for x in aobj.find(tag='ntfy_str')]) == 3
|
|
assert len([x for x in aobj.find(tag='ntfy_str_int')]) == 2
|
|
assert len([x for x in aobj.find(tag='ntfy_int')]) == 2
|
|
|
|
assert len([x for x in aobj.find(tag='ntfy_invalid')]) == 1
|
|
assert next(aobj.find(tag='ntfy_invalid')).priority == \
|
|
NtfyPriority.NORMAL
|