apprise/tests/test_plugin_ntfy.py

779 lines
24 KiB
Python

# BSD 2-Clause License
#
# Apprise - Push Notification Library.
# Copyright (c) 2025, Chris Caron <lead2gold@gmail.com>
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
#
# 1. Redistributions of source code must retain the above copyright notice,
# this list of conditions and the following disclaimer.
#
# 2. Redistributions in binary form must reproduce the above copyright notice,
# this list of conditions and the following disclaimer in the documentation
# and/or other materials provided with the distribution.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.
import json
# Disable logging for a cleaner testing output
import logging
import os
import re
from unittest import mock
from helpers import AppriseURLTester
import requests
import apprise
from apprise.plugins.ntfy import NotifyNtfy, NtfyPriority
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,
# Our expected url(privacy=True) startswith() response:
"privacy_url": "ntfy://user@localhost/topic",
},
),
# 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,
},
),
# No images
(
"ntfy://localhost/topic1/?image=False",
{
"instance": NotifyNtfy,
"requests_response_text": GOOD_RESPONSE_TEXT,
},
),
# Over-ride Image Path
(
"ntfy://localhost/topic1/?avatar_url=ttp://localhost/test.jpg",
{
"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,
},
),
# Auth Token Types (tk_ gets detected as a auth=token)
(
"ntfy://tk_abcd123456@localhost/topic1",
{
"instance": NotifyNtfy,
"requests_response_text": GOOD_RESPONSE_TEXT,
# Our expected url(privacy=True) startswith() response:
"privacy_url": "ntfy://t...6@localhost/topic1",
},
),
# Force an auth token since lack of tk_ prevents auto-detection
(
"ntfy://abcd123456@localhost/topic1?auth=token",
{
"instance": NotifyNtfy,
"requests_response_text": GOOD_RESPONSE_TEXT,
# Our expected url(privacy=True) startswith() response:
"privacy_url": "ntfy://a...6@localhost/topic1",
},
),
# Force an auth token since lack of tk_ prevents auto-detection
(
"ntfy://:abcd123456@localhost/topic1?auth=token",
{
"instance": NotifyNtfy,
"requests_response_text": GOOD_RESPONSE_TEXT,
# Our expected url(privacy=True) startswith() response:
"privacy_url": "ntfy://a...6@localhost/topic1",
},
),
# Token detection already implied when token keyword is set
(
"ntfy://localhost/topic1?token=abc1234",
{
"instance": NotifyNtfy,
"requests_response_text": GOOD_RESPONSE_TEXT,
# Our expected url(privacy=True) startswith() response:
"privacy_url": "ntfy://a...4@localhost/topic1",
},
),
# Token enforced, but since a user/pass provided, only the pass is kept
(
"ntfy://user:token@localhost/topic1?auth=token",
{
"instance": NotifyNtfy,
"requests_response_text": GOOD_RESPONSE_TEXT,
# Our expected url(privacy=True) startswith() response:
"privacy_url": "ntfy://t...n@localhost/topic1",
},
),
# Token mode force, but there was no token provided
(
"ntfy://localhost/topic1?auth=token",
{
"instance": NotifyNtfy,
# We'll out-right fail to send the notification
"response": False,
# Our expected url(privacy=True) startswith() response:
"privacy_url": "ntfy://localhost/topic1",
},
),
# Priority
(
"ntfy://localhost/topic1/?priority=default",
{
"instance": NotifyNtfy,
"requests_response_text": GOOD_RESPONSE_TEXT,
# Our expected url(privacy=True) startswith() response:
"privacy_url": "ntfy://localhost/topic1",
},
),
# 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,
},
),
(
"ntfys://token@localhost/topic/?auth=invalid",
{
# Invalid Authentication type
"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 i/o exceptions with 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"
# Reset our mock object
mock_post.reset_mock()
# Markdown Support
results = NotifyNtfy.parse_url("ntfys://topic/?format=markdown")
assert isinstance(results, dict)
instance = NotifyNtfy(**results)
assert (
instance.notify(
body="body", title="title", notify_type=apprise.NotifyType.INFO
)
is True
)
assert mock_post.call_count == 1
assert mock_post.call_args_list[0][0][0] == "https://ntfy.sh"
assert "X-Markdown" in mock_post.call_args_list[0][1]["headers"]
@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(list(aobj.find(tag="min"))) == 3
for s in aobj.find(tag="min"):
assert s.priority == NtfyPriority.MIN
assert len(list(aobj.find(tag="max"))) == 4
for s in aobj.find(tag="max"):
assert s.priority == NtfyPriority.MAX
assert len(list(aobj.find(tag="ntfy_str"))) == 3
assert len(list(aobj.find(tag="ntfy_str_int"))) == 2
assert len(list(aobj.find(tag="ntfy_int"))) == 2
assert len(list(aobj.find(tag="ntfy_invalid"))) == 1
assert next(aobj.find(tag="ntfy_invalid")).priority == NtfyPriority.NORMAL
# A cloud reference without any identifiers; the ntfy:// (insecure mode)
# is not considered during the id generation as ntfys:// is always
# implied
results = NotifyNtfy.parse_url("ntfy://")
obj = NotifyNtfy(**results)
new_results = NotifyNtfy.parse_url(obj.url())
obj2 = NotifyNtfy(**new_results)
assert obj.url_id() == obj2.url_id()
@mock.patch("requests.post")
def test_plugin_ntfy_internationalized_urls(mock_post):
"""NotifyNtfy() Internationalized URL Support."""
# Prepare Mock return object
response = mock.Mock()
response.content = GOOD_RESPONSE_TEXT
response.status_code = requests.codes.ok
mock_post.return_value = response
# Our input
title = "My Title"
body = "My Body"
# Google Translate promised me this just says 'Apprise Example' (I hope
# this is the case 🙏). Below is a URL requiring encoding so that it
# can be correctly passed into an http header:
click = "https://通知の例"
# Prepare our object
obj = apprise.Apprise.instantiate(f"ntfy://ntfy.sh/topic1?click={click}")
# Send our notification
assert obj.notify(title=title, body=body)
assert mock_post.call_count == 1
assert mock_post.call_args_list[0][0][0] == "http://ntfy.sh"
# Verify that our International URL was correctly escaped
assert (
"https://%25E9%2580%259A%25E7%259F%25A5%25E3%2581%25AE%25E4%25BE%258B"
in mock_post.call_args_list[0][1]["headers"]["X-Click"]
)
# Validate that we did not obstruct our URL in anyway
assert apprise.Apprise.instantiate(obj.url()).url() == obj.url()
@mock.patch("requests.post")
def test_plugin_ntfy_message_to_attach(mock_post):
"""NotifyNtfy() large messages converted into attachments."""
# Prepare Mock return object
response = mock.Mock()
response.content = GOOD_RESPONSE_TEXT
response.status_code = requests.codes.ok
mock_post.return_value = response
# Create a very, very big message
title = "My Title"
body = "b" * NotifyNtfy.ntfy_json_upstream_size_limit
for fmt in apprise.NOTIFY_FORMATS:
# Prepare our object
obj = apprise.Apprise.instantiate(
f"ntfy://user:pass@localhost:8080/topic?format={fmt}"
)
# Our content will actually transfer as an attachment
assert obj.notify(title=title, body=body)
assert mock_post.call_count == 1
assert (
mock_post.call_args_list[0][0][0] == "http://localhost:8080/topic"
)
response = mock_post.call_args_list[0][1]
assert "data" in response
assert response["data"].decode("utf-8").startswith(title)
assert response["data"].decode("utf-8").endswith(body)
assert "params" in response
assert "filename" in response["params"]
# Our filename is automatically generated (with .txt)
assert re.match(
r"^[a-z0-9-]+\.txt$", response["params"]["filename"], re.I
)
# Reset our mock object
mock_post.reset_mock()