2018-07-01 18:40:56 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
2019-02-01 03:05:56 +01:00
|
|
|
# Copyright (C) 2019 Chris Caron <lead2gold@gmail.com>
|
|
|
|
# All rights reserved.
|
2018-07-01 18:40:56 +02:00
|
|
|
#
|
2019-02-01 03:05:56 +01:00
|
|
|
# This code is licensed under the MIT License.
|
2018-07-01 18:40:56 +02: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-07-01 18:40:56 +02: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-07-01 18:40:56 +02: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-07-01 18:40:56 +02:00
|
|
|
|
2021-11-25 21:20:22 +01:00
|
|
|
import pytest
|
2018-07-01 18:40:56 +02:00
|
|
|
import mock
|
|
|
|
import sys
|
2019-02-25 07:02:29 +01:00
|
|
|
import six
|
2018-07-01 18:40:56 +02:00
|
|
|
import types
|
|
|
|
import apprise
|
|
|
|
|
|
|
|
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-07-01 18:40:56 +02:00
|
|
|
|
2021-11-25 21:20:22 +01:00
|
|
|
@pytest.mark.skipif(sys.version_info.major <= 2, reason="Requires Python 3.x+")
|
|
|
|
@pytest.mark.skipif((
|
|
|
|
'win32api' in sys.modules or
|
|
|
|
'win32con' in sys.modules or
|
|
|
|
'win32gui' in sys.modules), reason="Requires non-windows platform")
|
|
|
|
def test_plugin_windows_mocked():
|
2018-07-01 18:40:56 +02:00
|
|
|
"""
|
2021-11-25 21:20:22 +01:00
|
|
|
NotifyWindows() General Checks (via non-Windows platform)
|
2018-07-01 18:40:56 +02:00
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
# We need to fake our windows environment for testing purposes
|
|
|
|
win32api_name = 'win32api'
|
|
|
|
win32api = types.ModuleType(win32api_name)
|
|
|
|
sys.modules[win32api_name] = win32api
|
|
|
|
win32api.GetModuleHandle = mock.Mock(
|
|
|
|
name=win32api_name + '.GetModuleHandle')
|
|
|
|
win32api.PostQuitMessage = mock.Mock(
|
|
|
|
name=win32api_name + '.PostQuitMessage')
|
|
|
|
|
|
|
|
win32con_name = 'win32con'
|
|
|
|
win32con = types.ModuleType(win32con_name)
|
|
|
|
sys.modules[win32con_name] = win32con
|
|
|
|
win32con.CW_USEDEFAULT = mock.Mock(name=win32con_name + '.CW_USEDEFAULT')
|
|
|
|
win32con.IDI_APPLICATION = mock.Mock(
|
|
|
|
name=win32con_name + '.IDI_APPLICATION')
|
|
|
|
win32con.IMAGE_ICON = mock.Mock(name=win32con_name + '.IMAGE_ICON')
|
|
|
|
win32con.LR_DEFAULTSIZE = 1
|
|
|
|
win32con.LR_LOADFROMFILE = 2
|
|
|
|
win32con.WM_DESTROY = mock.Mock(name=win32con_name + '.WM_DESTROY')
|
|
|
|
win32con.WM_USER = 0
|
|
|
|
win32con.WS_OVERLAPPED = 1
|
|
|
|
win32con.WS_SYSMENU = 2
|
|
|
|
|
|
|
|
win32gui_name = 'win32gui'
|
|
|
|
win32gui = types.ModuleType(win32gui_name)
|
|
|
|
sys.modules[win32gui_name] = win32gui
|
|
|
|
win32gui.CreateWindow = mock.Mock(name=win32gui_name + '.CreateWindow')
|
|
|
|
win32gui.DestroyWindow = mock.Mock(name=win32gui_name + '.DestroyWindow')
|
|
|
|
win32gui.LoadIcon = mock.Mock(name=win32gui_name + '.LoadIcon')
|
|
|
|
win32gui.LoadImage = mock.Mock(name=win32gui_name + '.LoadImage')
|
|
|
|
win32gui.NIF_ICON = 1
|
|
|
|
win32gui.NIF_INFO = mock.Mock(name=win32gui_name + '.NIF_INFO')
|
|
|
|
win32gui.NIF_MESSAGE = 2
|
|
|
|
win32gui.NIF_TIP = 4
|
|
|
|
win32gui.NIM_ADD = mock.Mock(name=win32gui_name + '.NIM_ADD')
|
|
|
|
win32gui.NIM_DELETE = mock.Mock(name=win32gui_name + '.NIM_DELETE')
|
|
|
|
win32gui.NIM_MODIFY = mock.Mock(name=win32gui_name + '.NIM_MODIFY')
|
|
|
|
win32gui.RegisterClass = mock.Mock(name=win32gui_name + '.RegisterClass')
|
|
|
|
win32gui.UnregisterClass = mock.Mock(
|
|
|
|
name=win32gui_name + '.UnregisterClass')
|
|
|
|
win32gui.Shell_NotifyIcon = mock.Mock(
|
|
|
|
name=win32gui_name + '.Shell_NotifyIcon')
|
|
|
|
win32gui.UpdateWindow = mock.Mock(name=win32gui_name + '.UpdateWindow')
|
|
|
|
win32gui.WNDCLASS = mock.Mock(name=win32gui_name + '.WNDCLASS')
|
|
|
|
|
|
|
|
# The following allows our mocked content to kick in. In python 3.x keys()
|
|
|
|
# returns an iterator, therefore we need to convert the keys() back into
|
|
|
|
# a list object to prevent from getting the error:
|
|
|
|
# "RuntimeError: dictionary changed size during iteration"
|
|
|
|
#
|
|
|
|
for mod in list(sys.modules.keys()):
|
|
|
|
if mod.startswith('apprise.'):
|
2022-08-06 18:56:19 +02:00
|
|
|
del sys.modules[mod]
|
2018-07-01 18:40:56 +02:00
|
|
|
reload(apprise)
|
|
|
|
|
|
|
|
# Create our instance
|
|
|
|
obj = apprise.Apprise.instantiate('windows://', suppress_exceptions=False)
|
|
|
|
obj.duration = 0
|
|
|
|
|
2019-02-13 23:48:09 +01:00
|
|
|
# Test URL functionality
|
2019-10-09 18:39:31 +02:00
|
|
|
assert isinstance(obj.url(), six.string_types) is True
|
2019-02-13 23:48:09 +01:00
|
|
|
|
2018-07-01 18:40:56 +02:00
|
|
|
# Check that it found our mocked environments
|
2021-11-25 21:20:22 +01:00
|
|
|
assert obj.enabled is True
|
2018-07-01 18:40:56 +02:00
|
|
|
|
|
|
|
# _on_destroy check
|
|
|
|
obj._on_destroy(0, '', 0, 0)
|
|
|
|
|
|
|
|
# test notifications
|
2019-10-09 18:39:31 +02:00
|
|
|
assert obj.notify(
|
|
|
|
title='title', body='body',
|
|
|
|
notify_type=apprise.NotifyType.INFO) is True
|
2018-07-01 18:40:56 +02:00
|
|
|
|
2019-03-27 04:06:20 +01:00
|
|
|
obj = apprise.Apprise.instantiate(
|
|
|
|
'windows://_/?image=True', suppress_exceptions=False)
|
|
|
|
obj.duration = 0
|
2019-10-09 18:39:31 +02:00
|
|
|
assert isinstance(obj.url(), six.string_types) is True
|
|
|
|
assert obj.notify(
|
|
|
|
title='title', body='body',
|
|
|
|
notify_type=apprise.NotifyType.INFO) is True
|
2019-03-27 04:06:20 +01:00
|
|
|
|
|
|
|
obj = apprise.Apprise.instantiate(
|
|
|
|
'windows://_/?image=False', suppress_exceptions=False)
|
|
|
|
obj.duration = 0
|
2019-10-09 18:39:31 +02:00
|
|
|
assert isinstance(obj.url(), six.string_types) is True
|
|
|
|
assert obj.notify(
|
|
|
|
title='title', body='body',
|
|
|
|
notify_type=apprise.NotifyType.INFO) is True
|
2019-03-27 04:06:20 +01:00
|
|
|
|
|
|
|
obj = apprise.Apprise.instantiate(
|
|
|
|
'windows://_/?duration=1', suppress_exceptions=False)
|
2019-10-09 18:39:31 +02:00
|
|
|
assert isinstance(obj.url(), six.string_types) is True
|
2021-11-25 21:20:22 +01:00
|
|
|
# loads okay
|
|
|
|
assert obj.duration == 1
|
2019-10-09 18:39:31 +02:00
|
|
|
assert obj.notify(
|
|
|
|
title='title', body='body',
|
|
|
|
notify_type=apprise.NotifyType.INFO) is True
|
2019-03-27 04:06:20 +01:00
|
|
|
|
|
|
|
obj = apprise.Apprise.instantiate(
|
|
|
|
'windows://_/?duration=invalid', suppress_exceptions=False)
|
|
|
|
# Falls back to default
|
|
|
|
assert obj.duration == obj.default_popup_duration_sec
|
|
|
|
|
|
|
|
obj = apprise.Apprise.instantiate(
|
|
|
|
'windows://_/?duration=-1', suppress_exceptions=False)
|
|
|
|
# Falls back to default
|
|
|
|
assert obj.duration == obj.default_popup_duration_sec
|
|
|
|
|
|
|
|
obj = apprise.Apprise.instantiate(
|
|
|
|
'windows://_/?duration=0', suppress_exceptions=False)
|
|
|
|
# Falls back to default
|
|
|
|
assert obj.duration == obj.default_popup_duration_sec
|
|
|
|
|
2021-11-25 21:20:22 +01:00
|
|
|
# To avoid slowdowns (for testing), turn it to zero for now
|
|
|
|
obj.duration = 0
|
|
|
|
|
2018-07-01 18:40:56 +02:00
|
|
|
# Test our loading of our icon exception; it will still allow the
|
|
|
|
# notification to be sent
|
|
|
|
win32gui.LoadImage.side_effect = AttributeError
|
2019-10-09 18:39:31 +02:00
|
|
|
assert obj.notify(
|
|
|
|
title='title', body='body',
|
|
|
|
notify_type=apprise.NotifyType.INFO) is True
|
2018-07-01 18:40:56 +02:00
|
|
|
# Undo our change
|
|
|
|
win32gui.LoadImage.side_effect = None
|
|
|
|
|
|
|
|
# Test our global exception handling
|
|
|
|
win32gui.UpdateWindow.side_effect = AttributeError
|
2019-10-09 18:39:31 +02:00
|
|
|
assert obj.notify(
|
|
|
|
title='title', body='body',
|
|
|
|
notify_type=apprise.NotifyType.INFO) is False
|
2018-07-01 18:40:56 +02:00
|
|
|
# Undo our change
|
|
|
|
win32gui.UpdateWindow.side_effect = None
|
|
|
|
|
|
|
|
# Toggle our testing for when we can't send notifications because the
|
|
|
|
# package has been made unavailable to us
|
2021-11-25 21:20:22 +01:00
|
|
|
obj.enabled = False
|
|
|
|
assert obj.notify(
|
|
|
|
title='title', body='body',
|
|
|
|
notify_type=apprise.NotifyType.INFO) is False
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.skipif(
|
|
|
|
'win32api' not in sys.modules and
|
|
|
|
'win32con' not in sys.modules and
|
|
|
|
'win32gui' not in sys.modules,
|
|
|
|
reason="Requires win32api, win32con, and win32gui")
|
|
|
|
@mock.patch('win32gui.UpdateWindow')
|
|
|
|
@mock.patch('win32gui.Shell_NotifyIcon')
|
|
|
|
@mock.patch('win32gui.LoadImage')
|
|
|
|
def test_plugin_windows_native(
|
|
|
|
mock_update_window, mock_loadimage, mock_notify):
|
|
|
|
"""
|
|
|
|
NotifyWindows() General Checks (via Windows platform)
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Create our instance
|
|
|
|
obj = apprise.Apprise.instantiate('windows://', suppress_exceptions=False)
|
|
|
|
obj.duration = 0
|
|
|
|
|
|
|
|
# Test URL functionality
|
|
|
|
assert isinstance(obj.url(), six.string_types) is True
|
|
|
|
|
|
|
|
# Check that it found our mocked environments
|
|
|
|
assert obj.enabled is True
|
|
|
|
|
|
|
|
# _on_destroy check
|
|
|
|
obj._on_destroy(0, '', 0, 0)
|
|
|
|
|
|
|
|
# test notifications
|
|
|
|
assert obj.notify(
|
|
|
|
title='title', body='body',
|
|
|
|
notify_type=apprise.NotifyType.INFO) is True
|
|
|
|
|
|
|
|
obj = apprise.Apprise.instantiate(
|
|
|
|
'windows://_/?image=True', suppress_exceptions=False)
|
|
|
|
obj.duration = 0
|
|
|
|
assert isinstance(obj.url(), six.string_types) is True
|
|
|
|
assert obj.notify(
|
|
|
|
title='title', body='body',
|
|
|
|
notify_type=apprise.NotifyType.INFO) is True
|
|
|
|
|
|
|
|
obj = apprise.Apprise.instantiate(
|
|
|
|
'windows://_/?image=False', suppress_exceptions=False)
|
|
|
|
obj.duration = 0
|
|
|
|
assert isinstance(obj.url(), six.string_types) is True
|
|
|
|
assert obj.notify(
|
|
|
|
title='title', body='body',
|
|
|
|
notify_type=apprise.NotifyType.INFO) is True
|
|
|
|
|
|
|
|
obj = apprise.Apprise.instantiate(
|
|
|
|
'windows://_/?duration=1', suppress_exceptions=False)
|
|
|
|
assert isinstance(obj.url(), six.string_types) is True
|
|
|
|
assert obj.notify(
|
|
|
|
title='title', body='body',
|
|
|
|
notify_type=apprise.NotifyType.INFO) is True
|
|
|
|
# loads okay
|
|
|
|
assert obj.duration == 1
|
|
|
|
|
|
|
|
obj = apprise.Apprise.instantiate(
|
|
|
|
'windows://_/?duration=invalid', suppress_exceptions=False)
|
|
|
|
# Falls back to default
|
|
|
|
assert obj.duration == obj.default_popup_duration_sec
|
|
|
|
|
|
|
|
obj = apprise.Apprise.instantiate(
|
|
|
|
'windows://_/?duration=-1', suppress_exceptions=False)
|
|
|
|
# Falls back to default
|
|
|
|
assert obj.duration == obj.default_popup_duration_sec
|
|
|
|
|
|
|
|
obj = apprise.Apprise.instantiate(
|
|
|
|
'windows://_/?duration=0', suppress_exceptions=False)
|
|
|
|
# Falls back to default
|
|
|
|
assert obj.duration == obj.default_popup_duration_sec
|
|
|
|
|
|
|
|
# To avoid slowdowns (for testing), turn it to zero for now
|
|
|
|
obj.duration = 0
|
|
|
|
|
|
|
|
# Test our loading of our icon exception; it will still allow the
|
|
|
|
# notification to be sent
|
|
|
|
mock_loadimage.side_effect = AttributeError
|
|
|
|
assert obj.notify(
|
|
|
|
title='title', body='body',
|
|
|
|
notify_type=apprise.NotifyType.INFO) is True
|
|
|
|
# Undo our change
|
|
|
|
mock_loadimage.side_effect = None
|
|
|
|
|
|
|
|
# Test our global exception handling
|
|
|
|
mock_update_window.side_effect = AttributeError
|
|
|
|
assert obj.notify(
|
|
|
|
title='title', body='body',
|
|
|
|
notify_type=apprise.NotifyType.INFO) is False
|
|
|
|
# Undo our change
|
|
|
|
mock_update_window.side_effect = None
|
|
|
|
|
|
|
|
# Toggle our testing for when we can't send notifications because the
|
|
|
|
# package has been made unavailable to us
|
|
|
|
obj.enabled = False
|
2019-10-09 18:39:31 +02:00
|
|
|
assert obj.notify(
|
|
|
|
title='title', body='body',
|
|
|
|
notify_type=apprise.NotifyType.INFO) is False
|