2019-05-30 02:07:05 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
2023-02-09 09:54:55 +01:00
|
|
|
# BSD 3-Clause License
|
2019-05-30 02:07:05 +02:00
|
|
|
#
|
2023-02-09 09:54:55 +01:00
|
|
|
# Apprise - Push Notification Library.
|
|
|
|
# Copyright (c) 2023, Chris Caron <lead2gold@gmail.com>
|
2019-05-30 02:07:05 +02:00
|
|
|
#
|
2023-02-09 09:54:55 +01:00
|
|
|
# Redistribution and use in source and binary forms, with or without
|
|
|
|
# modification, are permitted provided that the following conditions are met:
|
2019-05-30 02:07:05 +02:00
|
|
|
#
|
2023-02-09 09:54:55 +01:00
|
|
|
# 1. Redistributions of source code must retain the above copyright notice,
|
|
|
|
# this list of conditions and the following disclaimer.
|
2019-05-30 02:07:05 +02:00
|
|
|
#
|
2023-02-09 09:54:55 +01:00
|
|
|
# 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.
|
2019-05-30 02:07:05 +02:00
|
|
|
#
|
2023-02-09 09:54:55 +01:00
|
|
|
# 3. Neither the name of the copyright holder nor the names of its
|
|
|
|
# contributors may be used to endorse or promote products derived from
|
|
|
|
# this software without specific prior written permission.
|
|
|
|
#
|
|
|
|
# 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.
|
2019-05-30 02:07:05 +02:00
|
|
|
|
2019-06-19 02:29:03 +02:00
|
|
|
import os
|
2022-10-30 21:31:57 +01:00
|
|
|
import sys
|
2022-10-08 02:28:36 +02:00
|
|
|
from unittest import mock
|
2022-09-01 02:05:40 +02:00
|
|
|
|
2019-05-30 02:07:05 +02:00
|
|
|
import ctypes
|
2022-10-30 21:31:57 +01:00
|
|
|
import pytest
|
2019-05-30 02:07:05 +02:00
|
|
|
|
|
|
|
from apprise import AppriseLocale
|
2019-06-05 04:13:15 +02:00
|
|
|
from apprise.utils import environ
|
2022-10-08 02:28:36 +02:00
|
|
|
from importlib import reload
|
2019-05-30 02:07:05 +02:00
|
|
|
|
|
|
|
|
|
|
|
# Disable logging for a cleaner testing output
|
|
|
|
import logging
|
|
|
|
logging.disable(logging.CRITICAL)
|
|
|
|
|
|
|
|
|
|
|
|
@mock.patch('gettext.install')
|
|
|
|
def test_apprise_locale(mock_gettext_install):
|
|
|
|
"""
|
|
|
|
API: Test apprise locale object
|
|
|
|
"""
|
|
|
|
lazytrans = AppriseLocale.LazyTranslation('Token')
|
|
|
|
assert str(lazytrans) == 'Token'
|
|
|
|
|
|
|
|
|
|
|
|
@mock.patch('gettext.install')
|
|
|
|
def test_gettext_init(mock_gettext_install):
|
|
|
|
"""
|
|
|
|
API: Mock Gettext init
|
|
|
|
"""
|
|
|
|
mock_gettext_install.side_effect = ImportError()
|
|
|
|
# Test our fall back to not supporting translations
|
|
|
|
reload(AppriseLocale)
|
|
|
|
|
|
|
|
# Objects can still be created
|
|
|
|
al = AppriseLocale.AppriseLocale()
|
|
|
|
|
|
|
|
with al.lang_at('en'):
|
|
|
|
# functions still behave as normal
|
|
|
|
pass
|
|
|
|
|
|
|
|
# restore the object
|
|
|
|
mock_gettext_install.side_effect = None
|
|
|
|
reload(AppriseLocale)
|
|
|
|
|
|
|
|
|
|
|
|
@mock.patch('gettext.translation')
|
|
|
|
def test_gettext_translations(mock_gettext_trans):
|
|
|
|
"""
|
|
|
|
API: Apprise() Gettext translations
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
mock_gettext_trans.side_effect = IOError()
|
|
|
|
|
|
|
|
# This throws internally but we handle it gracefully
|
|
|
|
al = AppriseLocale.AppriseLocale()
|
|
|
|
|
|
|
|
with al.lang_at('en'):
|
|
|
|
# functions still behave as normal
|
|
|
|
pass
|
|
|
|
|
|
|
|
# This throws internally but we handle it gracefully
|
|
|
|
AppriseLocale.AppriseLocale(language="fr")
|
|
|
|
|
|
|
|
|
|
|
|
@mock.patch('gettext.translation')
|
|
|
|
def test_gettext_installs(mock_gettext_trans):
|
|
|
|
"""
|
|
|
|
API: Apprise() Gettext install
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
mock_lang = mock.Mock()
|
|
|
|
mock_lang.install.return_value = True
|
|
|
|
mock_gettext_trans.return_value = mock_lang
|
|
|
|
|
|
|
|
# This throws internally but we handle it gracefully
|
|
|
|
al = AppriseLocale.AppriseLocale()
|
|
|
|
|
|
|
|
with al.lang_at('en'):
|
|
|
|
# functions still behave as normal
|
|
|
|
pass
|
|
|
|
|
|
|
|
# This throws internally but we handle it gracefully
|
|
|
|
AppriseLocale.AppriseLocale(language="fr")
|
|
|
|
|
|
|
|
# Force a few different languages
|
|
|
|
al._gtobjs['en'] = mock_lang
|
|
|
|
al._gtobjs['es'] = mock_lang
|
|
|
|
al.lang = 'en'
|
|
|
|
|
|
|
|
with al.lang_at('en'):
|
|
|
|
# functions still behave as normal
|
|
|
|
pass
|
|
|
|
|
|
|
|
with al.lang_at('es'):
|
|
|
|
# functions still behave as normal
|
|
|
|
pass
|
|
|
|
|
|
|
|
with al.lang_at('fr'):
|
|
|
|
# functions still behave as normal
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2019-06-05 04:13:15 +02:00
|
|
|
def test_detect_language_windows_users():
|
2019-05-30 02:07:05 +02:00
|
|
|
"""
|
|
|
|
API: Apprise() Detect language
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
2022-10-30 21:31:57 +01:00
|
|
|
if hasattr(ctypes, 'windll'):
|
|
|
|
from ctypes import windll
|
|
|
|
else:
|
2019-05-30 02:07:05 +02:00
|
|
|
windll = mock.Mock()
|
|
|
|
# 4105 = en_CA
|
|
|
|
windll.kernel32.GetUserDefaultUILanguage.return_value = 4105
|
|
|
|
setattr(ctypes, 'windll', windll)
|
|
|
|
|
|
|
|
# The below accesses the windows fallback code
|
2019-06-05 04:13:15 +02:00
|
|
|
with environ('LANG', 'LANGUAGE', 'LC_ALL', 'LC_CTYPE', LANG="en_CA"):
|
|
|
|
assert AppriseLocale.AppriseLocale.detect_language() == 'en'
|
2019-05-30 02:07:05 +02:00
|
|
|
|
|
|
|
assert AppriseLocale.AppriseLocale\
|
|
|
|
.detect_language(detect_fallback=False) is None
|
|
|
|
|
2019-06-05 04:13:15 +02:00
|
|
|
# 0 = IndexError
|
|
|
|
windll.kernel32.GetUserDefaultUILanguage.return_value = 0
|
|
|
|
setattr(ctypes, 'windll', windll)
|
|
|
|
with environ('LANG', 'LC_ALL', 'LC_CTYPE', LANGUAGE="en_CA"):
|
|
|
|
assert AppriseLocale.AppriseLocale.detect_language() == 'en'
|
|
|
|
|
2022-10-30 21:31:57 +01:00
|
|
|
|
|
|
|
@pytest.mark.skipif(sys.platform == "win32", reason="Does not work on Windows")
|
|
|
|
def test_detect_language_windows_users_croaks_please_review():
|
|
|
|
"""
|
|
|
|
When enabling CI testing on Windows, those tests did not produce the
|
|
|
|
correct results. They may want to be reviewed.
|
|
|
|
"""
|
|
|
|
|
2019-06-05 04:13:15 +02:00
|
|
|
# The below accesses the windows fallback code and fail
|
2022-10-30 21:31:57 +01:00
|
|
|
# then it will resort to the environment variables.
|
2019-06-05 04:13:15 +02:00
|
|
|
with environ('LANG', 'LANGUAGE', 'LC_ALL', 'LC_CTYPE'):
|
|
|
|
# Language can't be detected
|
|
|
|
assert AppriseLocale.AppriseLocale.detect_language() is None
|
|
|
|
|
2022-10-30 21:31:57 +01:00
|
|
|
# Detect French language.
|
2019-06-05 04:13:15 +02:00
|
|
|
with environ('LANGUAGE', 'LC_ALL', 'LC_CTYPE', LANG="fr_CA"):
|
|
|
|
assert AppriseLocale.AppriseLocale.detect_language() == 'fr'
|
|
|
|
|
2022-10-08 02:28:36 +02:00
|
|
|
# The following unsets all environment variables and sets LC_CTYPE
|
2019-06-19 02:29:03 +02:00
|
|
|
# This was causing Python 2.7 to internally parse UTF-8 as an invalid
|
2022-10-08 02:28:36 +02:00
|
|
|
# locale and throw an uncaught ValueError; Python v2 support has been
|
|
|
|
# dropped, but just to ensure this issue does not come back, we keep
|
|
|
|
# this test:
|
2019-06-19 02:29:03 +02:00
|
|
|
with environ(*list(os.environ.keys()), LC_CTYPE="UTF-8"):
|
|
|
|
assert AppriseLocale.AppriseLocale.detect_language() is None
|
|
|
|
|
|
|
|
# Test with absolutely no environment variables what-so-ever
|
|
|
|
with environ(*list(os.environ.keys())):
|
|
|
|
assert AppriseLocale.AppriseLocale.detect_language() is None
|
|
|
|
|
2019-06-05 04:13:15 +02:00
|
|
|
|
2022-10-30 21:31:57 +01:00
|
|
|
@pytest.mark.skipif(sys.platform == "win32", reason="Does not work on Windows")
|
2019-06-05 04:13:15 +02:00
|
|
|
@mock.patch('locale.getdefaultlocale')
|
|
|
|
def test_detect_language_defaultlocale(mock_getlocale):
|
|
|
|
"""
|
|
|
|
API: Apprise() Default locale detection
|
|
|
|
|
|
|
|
"""
|
2019-05-30 02:07:05 +02:00
|
|
|
# Handle case where getdefaultlocale() can't be detected
|
|
|
|
mock_getlocale.return_value = None
|
|
|
|
assert AppriseLocale.AppriseLocale.detect_language() is None
|
|
|
|
|
|
|
|
# if detect_language and windows env fail us, then we don't
|
|
|
|
# set up a default language on first load
|
|
|
|
AppriseLocale.AppriseLocale()
|