2018-03-14 00:26:51 +01:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
2019-02-01 03:05:56 +01:00
|
|
|
# Copyright (C) 2019 Chris Caron <lead2gold@gmail.com>
|
|
|
|
# All rights reserved.
|
2018-03-14 00:26:51 +01:00
|
|
|
#
|
2019-02-01 03:05:56 +01:00
|
|
|
# This code is licensed under the MIT License.
|
2018-03-14 00:26:51 +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 :
|
2018-03-14 00:26:51 +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.
|
2018-03-14 00:26:51 +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.
|
2018-03-14 00:26:51 +01:00
|
|
|
from __future__ import print_function
|
2019-09-30 00:19:55 +02:00
|
|
|
import re
|
2019-05-14 22:54:45 +02:00
|
|
|
import mock
|
2021-01-31 23:15:18 +01:00
|
|
|
import requests
|
|
|
|
import json
|
2019-11-10 07:10:03 +01:00
|
|
|
from os.path import dirname
|
|
|
|
from os.path import join
|
2018-03-14 00:26:51 +01:00
|
|
|
from apprise import cli
|
|
|
|
from apprise import NotifyBase
|
|
|
|
from click.testing import CliRunner
|
2019-02-25 07:02:29 +01:00
|
|
|
from apprise.plugins import SCHEMA_MAP
|
2020-12-26 23:36:33 +01:00
|
|
|
from apprise.utils import environ
|
2021-11-25 21:20:22 +01:00
|
|
|
from apprise.plugins import __load_matrix
|
|
|
|
from apprise.plugins import __reset_matrix
|
2020-12-26 23:36:33 +01:00
|
|
|
|
2018-03-14 00:26:51 +01:00
|
|
|
|
2019-05-14 22:54:45 +02:00
|
|
|
try:
|
|
|
|
# Python v3.4+
|
|
|
|
from importlib import reload
|
|
|
|
except ImportError:
|
|
|
|
try:
|
|
|
|
# Python v3.0-v3.3
|
|
|
|
from imp import reload
|
|
|
|
except ImportError:
|
|
|
|
# Python v2.7
|
|
|
|
pass
|
|
|
|
|
2019-02-25 07:02:29 +01:00
|
|
|
# Disable logging for a cleaner testing output
|
|
|
|
import logging
|
|
|
|
logging.disable(logging.CRITICAL)
|
2018-03-14 00:26:51 +01:00
|
|
|
|
2019-02-25 07:02:29 +01:00
|
|
|
|
2020-08-16 02:28:15 +02:00
|
|
|
def test_apprise_cli_nux_env(tmpdir):
|
2018-03-14 00:26:51 +01:00
|
|
|
"""
|
2020-08-16 02:28:15 +02:00
|
|
|
CLI: Nux Environment
|
2018-03-14 00:26:51 +01:00
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
class GoodNotification(NotifyBase):
|
2019-02-25 07:02:29 +01:00
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super(GoodNotification, self).__init__(*args, **kwargs)
|
2018-03-14 00:26:51 +01:00
|
|
|
|
|
|
|
def notify(self, **kwargs):
|
|
|
|
# Pretend everything is okay
|
|
|
|
return True
|
|
|
|
|
2019-09-30 00:19:55 +02:00
|
|
|
def url(self, *args, **kwargs):
|
2019-04-05 05:50:30 +02:00
|
|
|
# Support url()
|
2019-09-30 00:19:55 +02:00
|
|
|
return 'good://'
|
2019-04-05 05:50:30 +02:00
|
|
|
|
2018-03-14 00:26:51 +01:00
|
|
|
class BadNotification(NotifyBase):
|
2019-02-25 07:02:29 +01:00
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super(BadNotification, self).__init__(*args, **kwargs)
|
2018-03-14 00:26:51 +01:00
|
|
|
|
|
|
|
def notify(self, **kwargs):
|
2019-09-30 00:19:55 +02:00
|
|
|
# Force a notification failure
|
2018-03-14 00:26:51 +01:00
|
|
|
return False
|
|
|
|
|
2019-09-30 00:19:55 +02:00
|
|
|
def url(self, *args, **kwargs):
|
2019-04-05 05:50:30 +02:00
|
|
|
# Support url()
|
2019-09-30 00:19:55 +02:00
|
|
|
return 'bad://'
|
2019-04-05 05:50:30 +02:00
|
|
|
|
2018-03-14 00:26:51 +01:00
|
|
|
# Set up our notification types
|
|
|
|
SCHEMA_MAP['good'] = GoodNotification
|
|
|
|
SCHEMA_MAP['bad'] = BadNotification
|
|
|
|
|
|
|
|
runner = CliRunner()
|
|
|
|
result = runner.invoke(cli.main)
|
|
|
|
# no servers specified; we return 1 (non-zero)
|
|
|
|
assert result.exit_code == 1
|
|
|
|
|
|
|
|
result = runner.invoke(cli.main, ['-v'])
|
|
|
|
assert result.exit_code == 1
|
|
|
|
|
|
|
|
result = runner.invoke(cli.main, ['-vv'])
|
|
|
|
assert result.exit_code == 1
|
|
|
|
|
|
|
|
result = runner.invoke(cli.main, ['-vvv'])
|
|
|
|
assert result.exit_code == 1
|
|
|
|
|
2019-03-10 15:51:36 +01:00
|
|
|
result = runner.invoke(cli.main, ['-vvvv'])
|
|
|
|
assert result.exit_code == 1
|
|
|
|
|
2019-02-25 07:02:29 +01:00
|
|
|
# Display version information and exit
|
|
|
|
result = runner.invoke(cli.main, ['-V'])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
2018-03-14 00:26:51 +01:00
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-t', 'test title',
|
|
|
|
'-b', 'test body',
|
|
|
|
'good://localhost',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
2021-01-31 23:15:18 +01:00
|
|
|
with mock.patch('requests.post') as mock_post:
|
|
|
|
# Prepare Mock
|
|
|
|
mock_post.return_value = requests.Request()
|
|
|
|
mock_post.return_value.status_code = requests.codes.ok
|
|
|
|
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-t', 'test title',
|
|
|
|
'-b', 'test body\\nsNewLine',
|
|
|
|
# Test using interpret escapes
|
|
|
|
'-e',
|
|
|
|
# Use our JSON query
|
|
|
|
'json://localhost',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# Test our call count
|
|
|
|
assert mock_post.call_count == 1
|
|
|
|
|
|
|
|
# Our string is now escaped correctly
|
|
|
|
json.loads(mock_post.call_args_list[0][1]['data'])\
|
|
|
|
.get('message', '') == 'test body\nsNewLine'
|
|
|
|
|
|
|
|
# Reset
|
|
|
|
mock_post.reset_mock()
|
|
|
|
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-t', 'test title',
|
|
|
|
'-b', 'test body\\nsNewLine',
|
|
|
|
# No -e switch at all (so we don't escape the above)
|
|
|
|
# Use our JSON query
|
|
|
|
'json://localhost',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# Test our call count
|
|
|
|
assert mock_post.call_count == 1
|
|
|
|
|
|
|
|
# Our string is now escaped correctly
|
|
|
|
json.loads(mock_post.call_args_list[0][1]['data'])\
|
|
|
|
.get('message', '') == 'test body\\nsNewLine'
|
|
|
|
|
2020-08-16 02:28:15 +02:00
|
|
|
# Run in synchronous mode
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-t', 'test title',
|
|
|
|
'-b', 'test body',
|
|
|
|
'good://localhost',
|
|
|
|
'--disable-async',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# Test Debug Mode (--debug)
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-t', 'test title',
|
|
|
|
'-b', 'test body',
|
|
|
|
'good://localhost',
|
|
|
|
'--debug',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# Test Debug Mode (-D)
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-t', 'test title',
|
|
|
|
'-b', 'test body',
|
|
|
|
'good://localhost',
|
|
|
|
'-D',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
2018-03-14 00:26:51 +01:00
|
|
|
result = runner.invoke(cli.main, [
|
2018-09-06 03:27:52 +02:00
|
|
|
'-t', 'test title',
|
|
|
|
'good://localhost',
|
|
|
|
], input='test stdin body\n')
|
2018-03-14 00:26:51 +01:00
|
|
|
assert result.exit_code == 0
|
|
|
|
|
2020-08-16 02:28:15 +02:00
|
|
|
# Run in synchronous mode
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-t', 'test title',
|
|
|
|
'good://localhost',
|
|
|
|
'--disable-async',
|
|
|
|
], input='test stdin body\n')
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-t', 'test title',
|
|
|
|
'-b', 'test body',
|
|
|
|
'bad://localhost',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 1
|
|
|
|
|
|
|
|
# Run in synchronous mode
|
2018-03-14 00:26:51 +01:00
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-t', 'test title',
|
|
|
|
'-b', 'test body',
|
|
|
|
'bad://localhost',
|
2020-08-16 02:28:15 +02:00
|
|
|
'-Da',
|
2018-03-14 00:26:51 +01:00
|
|
|
])
|
|
|
|
assert result.exit_code == 1
|
2019-02-25 07:02:29 +01:00
|
|
|
|
2019-09-30 00:19:55 +02:00
|
|
|
# Testing with the --dry-run flag reveals a successful response since we
|
|
|
|
# don't actually execute the bad:// notification; we only display it
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-t', 'test title',
|
|
|
|
'-b', 'test body',
|
|
|
|
'bad://localhost',
|
|
|
|
'--dry-run',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
2019-02-25 07:02:29 +01:00
|
|
|
# Write a simple text based configuration file
|
|
|
|
t = tmpdir.mkdir("apprise-obj").join("apprise")
|
|
|
|
buf = """
|
2020-08-25 23:54:31 +02:00
|
|
|
# Include ourselves
|
|
|
|
include {}
|
|
|
|
|
2019-02-25 07:02:29 +01:00
|
|
|
taga,tagb=good://localhost
|
|
|
|
tagc=good://nuxref.com
|
2020-08-25 23:54:31 +02:00
|
|
|
""".format(str(t))
|
2019-02-25 07:02:29 +01:00
|
|
|
t.write(buf)
|
|
|
|
|
2019-09-30 00:19:55 +02:00
|
|
|
# This will read our configuration and not send any notices at all
|
|
|
|
# because we assigned tags to all of our urls and didn't identify
|
|
|
|
# a specific match below.
|
2020-08-25 23:54:31 +02:00
|
|
|
|
|
|
|
# 'include' reference in configuration file would have included the file a
|
|
|
|
# second time (since recursion default is 1).
|
2019-02-25 07:02:29 +01:00
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'test config',
|
|
|
|
'--config', str(t),
|
|
|
|
])
|
2020-08-25 23:54:31 +02:00
|
|
|
# Even when recursion take place, tags are all honored
|
|
|
|
# so 2 is returned because nothing was notified
|
|
|
|
assert result.exit_code == 3
|
2019-02-25 07:02:29 +01:00
|
|
|
|
|
|
|
# This will send out 1 notification because our tag matches
|
|
|
|
# one of the entries above
|
|
|
|
# translation: has taga
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'has taga',
|
|
|
|
'--config', str(t),
|
|
|
|
'--tag', 'taga',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
2020-08-25 23:54:31 +02:00
|
|
|
# Test recursion
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-t', 'test title',
|
|
|
|
'-b', 'test body',
|
|
|
|
'--config', str(t),
|
|
|
|
'--tag', 'tagc',
|
|
|
|
# Invalid entry specified for recursion
|
|
|
|
'-R', 'invalid',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 2
|
|
|
|
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-t', 'test title',
|
|
|
|
'-b', 'test body',
|
|
|
|
'--config', str(t),
|
|
|
|
'--tag', 'tagc',
|
|
|
|
# missing entry specified for recursion
|
|
|
|
'--recursive-depth',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 2
|
|
|
|
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-t', 'test title',
|
|
|
|
'-b', 'test body',
|
|
|
|
'--config', str(t),
|
|
|
|
'--tag', 'tagc',
|
|
|
|
# Disable recursion (thus inclusion will be ignored)
|
|
|
|
'-R', '0',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# Test recursion
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-t', 'test title',
|
|
|
|
'-b', 'test body',
|
|
|
|
'--config', str(t),
|
|
|
|
'--tag', 'tagc',
|
|
|
|
# Recurse up to 5 times
|
|
|
|
'--recursion-depth', '5',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
2019-02-25 07:02:29 +01:00
|
|
|
# This will send out 2 notifications because by specifying 2 tag
|
|
|
|
# entries, we 'or' them together:
|
|
|
|
# translation: has taga or tagb or tagd
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'has taga OR tagc OR tagd',
|
|
|
|
'--config', str(t),
|
|
|
|
'--tag', 'taga',
|
|
|
|
'--tag', 'tagc',
|
|
|
|
'--tag', 'tagd',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
2019-05-14 22:54:45 +02:00
|
|
|
|
2019-09-30 00:19:55 +02:00
|
|
|
# Write a simple text based configuration file
|
|
|
|
t = tmpdir.mkdir("apprise-obj2").join("apprise-test2")
|
|
|
|
buf = """
|
|
|
|
good://localhost/1
|
|
|
|
good://localhost/2
|
|
|
|
good://localhost/3
|
|
|
|
good://localhost/4
|
|
|
|
good://localhost/5
|
|
|
|
myTag=good://localhost/6
|
|
|
|
"""
|
|
|
|
t.write(buf)
|
|
|
|
|
|
|
|
# This will read our configuration and send a notification to
|
|
|
|
# the first 5 entries in the list, but not the one that has
|
|
|
|
# the tag associated with it
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'test config',
|
|
|
|
'--config', str(t),
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
2020-07-29 02:51:05 +02:00
|
|
|
# Test our notification type switch (it defaults to info) so we want to
|
|
|
|
# try it as a different value. Should return without a problem
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', '# test config',
|
|
|
|
'--config', str(t),
|
|
|
|
'-n', 'success',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# Test our notification type switch when set to something unsupported
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'test config',
|
|
|
|
'--config', str(t),
|
|
|
|
'--notification-type', 'invalid',
|
|
|
|
])
|
2020-08-25 23:54:31 +02:00
|
|
|
# An error code of 2 is returned if invalid input is specified on the
|
|
|
|
# command line
|
|
|
|
assert result.exit_code == 2
|
2020-07-29 02:51:05 +02:00
|
|
|
|
|
|
|
# The notification type switch is case-insensitive
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'test config',
|
|
|
|
'--config', str(t),
|
|
|
|
'--notification-type', 'WARNING',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# Test our formatting switch (it defaults to text) so we want to try it as
|
|
|
|
# a different value. Should return without a problem
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', '# test config',
|
|
|
|
'--config', str(t),
|
|
|
|
'-i', 'markdown',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# Test our formatting switch when set to something unsupported
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'test config',
|
|
|
|
'--config', str(t),
|
|
|
|
'--input-format', 'invalid',
|
|
|
|
])
|
2020-08-25 23:54:31 +02:00
|
|
|
# An error code of 2 is returned if invalid input is specified on the
|
|
|
|
# command line
|
|
|
|
assert result.exit_code == 2
|
2020-07-29 02:51:05 +02:00
|
|
|
|
|
|
|
# The formatting switch is not case sensitive
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', '# test config',
|
|
|
|
'--config', str(t),
|
|
|
|
'--input-format', 'HTML',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
2019-09-30 00:19:55 +02:00
|
|
|
# As a way of ensuring we match the first 5 entries, we can run a
|
|
|
|
# --dry-run against the same result set above and verify the output
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'test config',
|
|
|
|
'--config', str(t),
|
|
|
|
'--dry-run',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
lines = re.split(r'[\r\n]', result.output.strip())
|
|
|
|
# 5 lines of all good:// entries matched
|
|
|
|
assert len(lines) == 5
|
|
|
|
# Verify we match against the remaining good:// entries
|
|
|
|
for i in range(0, 5):
|
|
|
|
assert lines[i].endswith('good://')
|
|
|
|
|
|
|
|
# This will fail because nothing matches mytag. It's case sensitive
|
|
|
|
# and we would only actually match against myTag
|
|
|
|
result = runner.invoke(cli.main, [
|
2020-12-26 23:36:33 +01:00
|
|
|
'-b', 'has mytag',
|
2019-09-30 00:19:55 +02:00
|
|
|
'--config', str(t),
|
|
|
|
'--tag', 'mytag',
|
|
|
|
])
|
2020-08-25 23:54:31 +02:00
|
|
|
assert result.exit_code == 3
|
2019-09-30 00:19:55 +02:00
|
|
|
|
|
|
|
# Same command as the one identified above except we set the --dry-run
|
|
|
|
# flag. This causes our list of matched results to be printed only.
|
|
|
|
# However, since we don't match anything; we still fail with a return code
|
|
|
|
# of 2.
|
|
|
|
result = runner.invoke(cli.main, [
|
2020-12-26 23:36:33 +01:00
|
|
|
'-b', 'has mytag',
|
2019-09-30 00:19:55 +02:00
|
|
|
'--config', str(t),
|
|
|
|
'--tag', 'mytag',
|
|
|
|
'--dry-run'
|
|
|
|
])
|
2020-08-25 23:54:31 +02:00
|
|
|
assert result.exit_code == 3
|
2019-09-30 00:19:55 +02:00
|
|
|
|
2019-11-10 07:10:03 +01:00
|
|
|
# Here is a case where we get what was expected; we also attach a file
|
2019-09-30 00:19:55 +02:00
|
|
|
result = runner.invoke(cli.main, [
|
2020-12-26 23:36:33 +01:00
|
|
|
'-b', 'has myTag',
|
2019-09-30 00:19:55 +02:00
|
|
|
'--config', str(t),
|
2019-11-10 07:10:03 +01:00
|
|
|
'--attach', join(dirname(__file__), 'var', 'apprise-test.gif'),
|
2019-09-30 00:19:55 +02:00
|
|
|
'--tag', 'myTag',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# Testing with the --dry-run flag reveals the same positive results
|
|
|
|
# because there was at least one match
|
|
|
|
result = runner.invoke(cli.main, [
|
2020-12-26 23:36:33 +01:00
|
|
|
'-b', 'has myTag',
|
2019-09-30 00:19:55 +02:00
|
|
|
'--config', str(t),
|
|
|
|
'--tag', 'myTag',
|
|
|
|
'--dry-run',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
2020-12-26 23:36:33 +01:00
|
|
|
#
|
|
|
|
# Test environment variables
|
|
|
|
#
|
|
|
|
# Write a simple text based configuration file
|
|
|
|
t2 = tmpdir.mkdir("apprise-obj-env").join("apprise")
|
|
|
|
buf = """
|
|
|
|
# A general one
|
|
|
|
good://localhost
|
|
|
|
|
|
|
|
# A failure (if we use the fail tag)
|
|
|
|
fail=bad://localhost
|
|
|
|
|
|
|
|
# A normal one tied to myTag
|
|
|
|
myTag=good://nuxref.com
|
|
|
|
"""
|
|
|
|
t2.write(buf)
|
|
|
|
|
|
|
|
with environ(APPRISE_URLS="good://localhost"):
|
|
|
|
# This will load okay because we defined the environment
|
|
|
|
# variable with a valid URL
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'test environment',
|
|
|
|
# Test that we ignore our tag
|
|
|
|
'--tag', 'mytag',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# Same action but without --tag
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'test environment',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
2020-12-31 16:16:03 +01:00
|
|
|
with mock.patch('apprise.cli.DEFAULT_SEARCH_PATHS', []):
|
|
|
|
with environ(APPRISE_URLS=" "):
|
2021-01-10 21:55:59 +01:00
|
|
|
# An empty string is not valid and therefore not loaded so the
|
|
|
|
# below fails. We override the DEFAULT_SEARCH_PATHS because we
|
|
|
|
# don't want to detect ones loaded on the machine running the unit
|
|
|
|
# tests
|
2020-12-31 16:16:03 +01:00
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'test environment',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 1
|
2020-12-26 23:36:33 +01:00
|
|
|
|
|
|
|
with environ(APPRISE_URLS="bad://localhost"):
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'test environment',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 1
|
|
|
|
|
|
|
|
# If we specify an inline URL, it will over-ride the environment
|
|
|
|
# variable
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-t', 'test title',
|
|
|
|
'-b', 'test body',
|
|
|
|
'good://localhost',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# A Config file also over-rides the environment variable if
|
|
|
|
# specified on the command line:
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'has myTag',
|
|
|
|
'--config', str(t2),
|
|
|
|
'--tag', 'myTag',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
with environ(APPRISE_CONFIG=str(t2)):
|
|
|
|
# Our configuration file will load from our environmment variable
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'has myTag',
|
|
|
|
'--tag', 'myTag',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
2020-12-31 16:16:03 +01:00
|
|
|
with mock.patch('apprise.cli.DEFAULT_SEARCH_PATHS', []):
|
|
|
|
with environ(APPRISE_CONFIG=" "):
|
|
|
|
# We will fail to send the notification as no path was
|
|
|
|
# specified.
|
|
|
|
# We override the DEFAULT_SEARCH_PATHS because we don't
|
|
|
|
# want to detect ones loaded on the machine running the unit tests
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'my message',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 1
|
2020-12-26 23:36:33 +01:00
|
|
|
|
|
|
|
with environ(APPRISE_CONFIG="garbage/file/path.yaml"):
|
|
|
|
# We will fail to send the notification as the path
|
|
|
|
# specified is not loadable
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'my message',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 1
|
|
|
|
|
|
|
|
# We can force an over-ride by specifying a config file on the
|
|
|
|
# command line options:
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'has myTag',
|
|
|
|
'--config', str(t2),
|
|
|
|
'--tag', 'myTag',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# Just a general test; if both the --config and urls are specified
|
|
|
|
# then the the urls trumps all
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'has myTag',
|
|
|
|
'--config', str(t2),
|
|
|
|
'good://localhost',
|
|
|
|
'--tag', 'fail',
|
|
|
|
])
|
|
|
|
# Tags are ignored, URL specified, so it trump config
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# we just repeat the test as a proof that it only executes
|
|
|
|
# the urls despite the fact the --config was specified
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'reads the url entry only',
|
|
|
|
'--config', str(t2),
|
|
|
|
'good://localhost',
|
|
|
|
'--tag', 'fail',
|
|
|
|
])
|
|
|
|
# Tags are ignored, URL specified, so it trump config
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# once agian, but we call bad://
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-b', 'reads the url entry only',
|
|
|
|
'--config', str(t2),
|
|
|
|
'bad://localhost',
|
|
|
|
'--tag', 'myTag',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 1
|
|
|
|
|
2021-10-30 23:00:05 +02:00
|
|
|
# Test Escaping:
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-e',
|
|
|
|
'-t', 'test\ntitle',
|
|
|
|
'-b', 'test\nbody',
|
|
|
|
'good://localhost',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# Test Escaping (without title)
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'--interpret-escapes',
|
|
|
|
'-b', 'test\nbody',
|
|
|
|
'good://localhost',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
2019-05-14 22:54:45 +02:00
|
|
|
|
2021-11-25 21:20:22 +01:00
|
|
|
def test_apprise_cli_details(tmpdir):
|
|
|
|
"""
|
|
|
|
API: Apprise() Disabled Plugin States
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
runner = CliRunner()
|
|
|
|
|
|
|
|
#
|
|
|
|
# Testing the printout of our details
|
|
|
|
# --details or -l
|
|
|
|
#
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'--details',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-l',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# Reset our matrix
|
|
|
|
__reset_matrix()
|
|
|
|
|
|
|
|
# This is a made up class that is just used to verify
|
|
|
|
class TestReq01Notification(NotifyBase):
|
|
|
|
"""
|
|
|
|
This class is used to test various requirement configurations
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Set some requirements
|
|
|
|
requirements = {
|
|
|
|
'packages_required': [
|
|
|
|
'cryptography <= 3.4',
|
|
|
|
'ultrasync',
|
|
|
|
],
|
|
|
|
'packages_recommended': 'django',
|
|
|
|
}
|
|
|
|
|
|
|
|
def url(self, **kwargs):
|
|
|
|
# Support URL
|
|
|
|
return ''
|
|
|
|
|
|
|
|
def send(self, **kwargs):
|
|
|
|
# Pretend everything is okay (so we don't break other tests)
|
|
|
|
return True
|
|
|
|
|
|
|
|
SCHEMA_MAP['req01'] = TestReq01Notification
|
|
|
|
|
|
|
|
# This is a made up class that is just used to verify
|
|
|
|
class TestReq02Notification(NotifyBase):
|
|
|
|
"""
|
|
|
|
This class is used to test various requirement configurations
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Just not enabled at all
|
|
|
|
enabled = False
|
|
|
|
|
|
|
|
# Set some requirements
|
|
|
|
requirements = {
|
|
|
|
# None and/or [] is implied, but jsut to show that the code won't
|
|
|
|
# crash if explicitly set this way:
|
|
|
|
'packages_required': None,
|
|
|
|
|
|
|
|
'packages_recommended': [
|
|
|
|
'cryptography <= 3.4',
|
|
|
|
]
|
|
|
|
}
|
|
|
|
|
|
|
|
def url(self, **kwargs):
|
|
|
|
# Support URL
|
|
|
|
return ''
|
|
|
|
|
|
|
|
def send(self, **kwargs):
|
|
|
|
# Pretend everything is okay (so we don't break other tests)
|
|
|
|
return True
|
|
|
|
|
|
|
|
SCHEMA_MAP['req02'] = TestReq02Notification
|
|
|
|
|
|
|
|
# This is a made up class that is just used to verify
|
|
|
|
class TestReq03Notification(NotifyBase):
|
|
|
|
"""
|
|
|
|
This class is used to test various requirement configurations
|
|
|
|
"""
|
|
|
|
|
2021-11-26 17:24:43 +01:00
|
|
|
# Set some requirements (but additionally include a details over-ride)
|
2021-11-25 21:20:22 +01:00
|
|
|
requirements = {
|
|
|
|
# We can over-ride the default details assigned to our plugin if
|
|
|
|
# specified
|
|
|
|
'details': _('some specified requirement details'),
|
|
|
|
|
|
|
|
# We can set a string value as well (it does not have to be a list)
|
|
|
|
'packages_recommended': 'cryptography <= 3.4'
|
|
|
|
}
|
|
|
|
|
|
|
|
def url(self, **kwargs):
|
|
|
|
# Support URL
|
|
|
|
return ''
|
|
|
|
|
|
|
|
def send(self, **kwargs):
|
|
|
|
# Pretend everything is okay (so we don't break other tests)
|
|
|
|
return True
|
|
|
|
|
|
|
|
SCHEMA_MAP['req03'] = TestReq03Notification
|
|
|
|
|
2021-11-26 17:24:43 +01:00
|
|
|
# This is a made up class that is just used to verify
|
|
|
|
class TestReq04Notification(NotifyBase):
|
|
|
|
"""
|
|
|
|
This class is used to test a case where our requirements is fixed
|
|
|
|
to a None
|
|
|
|
"""
|
|
|
|
|
|
|
|
# This is the same as saying there are no requirements
|
|
|
|
requirements = None
|
|
|
|
|
|
|
|
def url(self, **kwargs):
|
|
|
|
# Support URL
|
|
|
|
return ''
|
|
|
|
|
|
|
|
def send(self, **kwargs):
|
|
|
|
# Pretend everything is okay (so we don't break other tests)
|
|
|
|
return True
|
|
|
|
|
|
|
|
SCHEMA_MAP['req04'] = TestReq04Notification
|
|
|
|
|
|
|
|
# This is a made up class that is just used to verify
|
|
|
|
class TestReq05Notification(NotifyBase):
|
|
|
|
"""
|
|
|
|
This class is used to test a case where only packages_recommended
|
|
|
|
is identified
|
|
|
|
"""
|
|
|
|
|
|
|
|
requirements = {
|
|
|
|
'packages_recommended': 'cryptography <= 3.4'
|
|
|
|
}
|
|
|
|
|
|
|
|
def url(self, **kwargs):
|
|
|
|
# Support URL
|
|
|
|
return ''
|
|
|
|
|
|
|
|
def send(self, **kwargs):
|
|
|
|
# Pretend everything is okay (so we don't break other tests)
|
|
|
|
return True
|
|
|
|
|
|
|
|
SCHEMA_MAP['req05'] = TestReq04Notification
|
|
|
|
|
2021-11-25 21:20:22 +01:00
|
|
|
class TestDisabled01Notification(NotifyBase):
|
|
|
|
"""
|
|
|
|
This class is used to test a pre-disabled state
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Just flat out disable our service
|
|
|
|
enabled = False
|
|
|
|
|
|
|
|
# we'll use this as a key to make our service easier to find
|
|
|
|
# in the next part of the testing
|
|
|
|
service_name = 'na01'
|
|
|
|
|
|
|
|
def url(self, **kwargs):
|
|
|
|
# Support URL
|
|
|
|
return ''
|
|
|
|
|
|
|
|
def notify(self, **kwargs):
|
|
|
|
# Pretend everything is okay (so we don't break other tests)
|
|
|
|
return True
|
|
|
|
|
|
|
|
SCHEMA_MAP['na01'] = TestDisabled01Notification
|
|
|
|
|
|
|
|
class TestDisabled02Notification(NotifyBase):
|
|
|
|
"""
|
|
|
|
This class is used to test a post-disabled state
|
|
|
|
"""
|
|
|
|
|
|
|
|
# we'll use this as a key to make our service easier to find
|
|
|
|
# in the next part of the testing
|
|
|
|
service_name = 'na02'
|
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super(TestDisabled02Notification, self).__init__(**kwargs)
|
|
|
|
|
|
|
|
# enable state changes **AFTER** we initialize
|
|
|
|
self.enabled = False
|
|
|
|
|
|
|
|
def url(self, **kwargs):
|
|
|
|
# Support URL
|
|
|
|
return ''
|
|
|
|
|
|
|
|
def notify(self, **kwargs):
|
|
|
|
# Pretend everything is okay (so we don't break other tests)
|
|
|
|
return True
|
|
|
|
|
|
|
|
SCHEMA_MAP['na02'] = TestDisabled02Notification
|
|
|
|
|
|
|
|
# We'll add a good notification to our list
|
|
|
|
class TesEnabled01Notification(NotifyBase):
|
|
|
|
"""
|
|
|
|
This class is just a simple enabled one
|
|
|
|
"""
|
|
|
|
|
|
|
|
# we'll use this as a key to make our service easier to find
|
|
|
|
# in the next part of the testing
|
|
|
|
service_name = 'good'
|
|
|
|
|
|
|
|
def url(self, **kwargs):
|
|
|
|
# Support URL
|
|
|
|
return ''
|
|
|
|
|
|
|
|
def send(self, **kwargs):
|
|
|
|
# Pretend everything is okay (so we don't break other tests)
|
|
|
|
return True
|
|
|
|
|
|
|
|
SCHEMA_MAP['good'] = TesEnabled01Notification
|
|
|
|
|
|
|
|
# Verify that we can pass through all of our different details
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'--details',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
result = runner.invoke(cli.main, [
|
|
|
|
'-l',
|
|
|
|
])
|
|
|
|
assert result.exit_code == 0
|
|
|
|
|
|
|
|
# Reset our matrix
|
|
|
|
__reset_matrix()
|
|
|
|
__load_matrix()
|
|
|
|
|
|
|
|
|
2019-05-14 22:54:45 +02:00
|
|
|
@mock.patch('platform.system')
|
|
|
|
def test_apprise_cli_windows_env(mock_system):
|
|
|
|
"""
|
2020-08-16 02:28:15 +02:00
|
|
|
CLI: Windows Environment
|
2019-05-14 22:54:45 +02:00
|
|
|
|
|
|
|
"""
|
|
|
|
# Force a windows environment
|
|
|
|
mock_system.return_value = 'Windows'
|
|
|
|
|
|
|
|
# Reload our module
|
|
|
|
reload(cli)
|