Merge branch 'master' of gitlab.com:openlp/openlp

This commit is contained in:
Tim 2019-12-07 20:20:21 +00:00
commit 9eb645cc37
No known key found for this signature in database
GPG Key ID: 3D454289AF831A6D
16 changed files with 1292 additions and 1077 deletions

1
.gitignore vendored
View File

@ -25,6 +25,7 @@
.vscode .vscode
.eggs .eggs
.venv .venv
.mypy_cache
OpenLP.egg-info OpenLP.egg-info
\#*\# \#*\#
__pycache__ __pycache__

View File

@ -63,19 +63,6 @@ test-macos:
only: only:
- master@openlp/openlp - master@openlp/openlp
test-windows:
stage: test
tags:
- windows
script:
- C:\Users\raoul\GitLab-Runner\venv\Scripts\pytest.exe --color=no --disable-warnings --cov openlp
- mv .coverage windows.coverage
artifacts:
paths:
- windows.coverage
only:
- master@openlp/openlp
test-display: test-display:
stage: test stage: test
image: openlp/angular image: openlp/angular
@ -87,7 +74,7 @@ pages:
stage: deploy stage: deploy
image: openlp/debian image: openlp/debian
script: script:
- python3-coverage combine linux.coverage macos.coverage windows.coverage - python3-coverage combine linux.coverage macos.coverage
- fixpaths .coverage - fixpaths .coverage
- python3-coverage html - python3-coverage html
- mv htmlcov public - mv htmlcov public
@ -99,6 +86,5 @@ pages:
dependencies: dependencies:
- test-debian - test-debian
- test-macos - test-macos
- test-windows
only: only:
- master@openlp/openlp - master@openlp/openlp

View File

@ -385,17 +385,12 @@ def main():
application.setApplicationName('OpenLP') application.setApplicationName('OpenLP')
set_up_logging(AppLocation.get_directory(AppLocation.CacheDir)) set_up_logging(AppLocation.get_directory(AppLocation.CacheDir))
# Set the libvlc environment variable if we're frozen # Set the libvlc environment variable if we're frozen
if getattr(sys, 'frozen', False): if getattr(sys, 'frozen', False) and is_win():
if is_macosx(): # Path to libvlc and the plugins
vlc_lib = 'libvlc.dylib' os.environ['PYTHON_VLC_LIB_PATH'] = str(AppLocation.get_directory(AppLocation.AppDir) / 'vlc' / 'libvlc.dll')
elif is_win():
vlc_lib = 'libvlc.dll'
# Path to libvlc
os.environ['PYTHON_VLC_LIB_PATH'] = str(AppLocation.get_directory(AppLocation.AppDir) / 'vlc' / vlc_lib)
log.debug('VLC Path: {}'.format(os.environ['PYTHON_VLC_LIB_PATH']))
# Path to VLC directory containing VLC's "plugins" directory
os.environ['PYTHON_VLC_MODULE_PATH'] = str(AppLocation.get_directory(AppLocation.AppDir) / 'vlc') os.environ['PYTHON_VLC_MODULE_PATH'] = str(AppLocation.get_directory(AppLocation.AppDir) / 'vlc')
log.debug('VLC Path: {}'.format(os.environ['PYTHON_VLC_LIB_PATH'])) log.debug('VLC Path: {}'.format(os.environ['PYTHON_VLC_LIB_PATH']))
log.debug('VLC Plugins Path: {}'.format(os.environ['PYTHON_VLC_MODULE_PATH']))
# Initialise the Registry # Initialise the Registry
Registry.create() Registry.create()
Registry().register('application', application) Registry().register('application', application)

View File

@ -38,6 +38,13 @@ from PyQt5.QtCore import QCryptographicHash as QHash
from PyQt5.QtNetwork import QAbstractSocket, QHostAddress, QNetworkInterface from PyQt5.QtNetwork import QAbstractSocket, QHostAddress, QNetworkInterface
from chardet.universaldetector import UniversalDetector from chardet.universaldetector import UniversalDetector
try:
from distro import id as distro_id
except ImportError:
# The distro module is only valid for Linux, so if it doesn't exist, create a function that always returns False
def distro_id():
return False
log = logging.getLogger(__name__ + '.__init__') log = logging.getLogger(__name__ + '.__init__')
@ -212,13 +219,17 @@ def is_macosx():
return sys.platform.startswith('darwin') return sys.platform.startswith('darwin')
def is_linux(): def is_linux(distro=None):
""" """
Returns true if running on a system with a linux kernel e.g. Ubuntu, Debian, etc Returns true if running on a system with a linux kernel e.g. Ubuntu, Debian, etc
:param distro: If not None, check if running that Linux distro
:return: True if system is running a linux kernel false otherwise :return: True if system is running a linux kernel false otherwise
""" """
return sys.platform.startswith('linux') result = sys.platform.startswith('linux')
if result and distro:
result = result and distro == distro_id()
return result
def is_64bit_instance(): def is_64bit_instance():

View File

@ -34,6 +34,7 @@ from PyQt5 import QtCore
from openlp.core.state import State from openlp.core.state import State
from openlp.core.api.http import register_endpoint from openlp.core.api.http import register_endpoint
from openlp.core.common import is_linux, is_macosx
from openlp.core.common.i18n import translate from openlp.core.common.i18n import translate
from openlp.core.common.mixins import LogMixin, RegistryProperties from openlp.core.common.mixins import LogMixin, RegistryProperties
from openlp.core.common.registry import Registry, RegistryBase from openlp.core.common.registry import Registry, RegistryBase
@ -89,23 +90,35 @@ class MediaController(RegistryBase, LogMixin, RegistryProperties):
self.vlc_player = VlcPlayer(self) self.vlc_player = VlcPlayer(self)
State().add_service('mediacontroller', 0) State().add_service('mediacontroller', 0)
State().add_service('media_live', 0) State().add_service('media_live', 0)
getvlc = get_vlc() has_vlc = get_vlc()
if getvlc and pymediainfo_available: if has_vlc and pymediainfo_available:
State().update_pre_conditions('mediacontroller', True) State().update_pre_conditions('mediacontroller', True)
State().update_pre_conditions('media_live', True) State().update_pre_conditions('media_live', True)
else: else:
if hasattr(self.main_window, 'splash') and self.main_window.splash.isVisible(): if hasattr(self.main_window, 'splash') and self.main_window.splash.isVisible():
self.main_window.splash.hide() self.main_window.splash.hide()
text_vlc = translate('OpenLP.MediaController', generic_message = translate('OpenLP.MediaController',
'The media integration library is missing (python - vlc is not installed)') 'OpenLP requires the following libraries in order to show videos and other '
text_info = translate('OpenLP.MediaController', 'media, but they are not installed. Please install these libraries to enable '
'The media integration library is missing (python - pymediainfo is not installed)') 'media playback in OpenLP.')
if not getvlc and not pymediainfo_available: fedora_rpmfusion = translate('OpenLP.MediaController',
State().missing_text('media_live', "{text}\n{base}".format(text=text_vlc, base=text_info)) 'To install these libraries, you will need to enable the RPMFusion '
if getvlc and not pymediainfo_available: 'repository: https://rpmfusion.org/')
State().missing_text('media_live', "{text}".format(text=text_info)) message = ''
if not getvlc and pymediainfo_available: if is_macosx():
State().missing_text('media_live', "{text}".format(text=text_vlc)) message = translate('OpenLP.MediaController',
'macOS is missing VLC. Please download and install from the VLC web site: '
'https://www.videolan.org/vlc/')
else:
packages = []
if not has_vlc:
packages.append('python3-vlc')
if not pymediainfo_available:
packages.append('python3-pymediainfo')
message = generic_message + '\n\n' + ', '.join(packages)
if not has_vlc and is_linux(distro='fedora'):
message += '\n\n' + fedora_rpmfusion
State().missing_text('media_live', message)
return True return True
def bootstrap_post_set_up(self): def bootstrap_post_set_up(self):

View File

@ -34,7 +34,6 @@ from openlp.core.common.actions import ActionList, CategoryOrder
from openlp.core.common.i18n import UiStrings, translate from openlp.core.common.i18n import UiStrings, translate
from openlp.core.common.mixins import LogMixin, RegistryProperties from openlp.core.common.mixins import LogMixin, RegistryProperties
from openlp.core.common.registry import Registry, RegistryBase from openlp.core.common.registry import Registry, RegistryBase
from openlp.core.common.settings import Settings
from openlp.core.display.screens import ScreenList from openlp.core.display.screens import ScreenList
from openlp.core.display.window import DisplayWindow from openlp.core.display.window import DisplayWindow
from openlp.core.lib import ServiceItemAction, image_to_byte from openlp.core.lib import ServiceItemAction, image_to_byte
@ -324,7 +323,7 @@ class SlideController(QtWidgets.QWidget, LogMixin, RegistryProperties):
self.play_slides_once = create_action(self, 'playSlidesOnce', text=UiStrings().PlaySlidesToEnd, self.play_slides_once = create_action(self, 'playSlidesOnce', text=UiStrings().PlaySlidesToEnd,
icon=UiIcons().clock, checked=False, can_shortcuts=True, icon=UiIcons().clock, checked=False, can_shortcuts=True,
category=self.category, triggers=self.on_play_slides_once) category=self.category, triggers=self.on_play_slides_once)
if Settings().value(self.main_window.advanced_settings_section + '/slide limits') == SlideLimits.Wrap: if self.settings.value(self.main_window.advanced_settings_section + '/slide limits') == SlideLimits.Wrap:
self.play_slides_menu.setDefaultAction(self.play_slides_loop) self.play_slides_menu.setDefaultAction(self.play_slides_loop)
else: else:
self.play_slides_menu.setDefaultAction(self.play_slides_once) self.play_slides_menu.setDefaultAction(self.play_slides_once)
@ -697,13 +696,13 @@ class SlideController(QtWidgets.QWidget, LogMixin, RegistryProperties):
""" """
Adjusts the value of the ``delay_spin_box`` to the given one. Adjusts the value of the ``delay_spin_box`` to the given one.
""" """
self.delay_spin_box.setValue(Settings().value('core/loop delay')) self.delay_spin_box.setValue(self.settings.value('core/loop delay'))
def update_slide_limits(self): def update_slide_limits(self):
""" """
Updates the Slide Limits variable from the settings. Updates the Slide Limits variable from the settings.
""" """
self.slide_limits = Settings().value(self.main_window.advanced_settings_section + '/slide limits') self.slide_limits = self.settings.value(self.main_window.advanced_settings_section + '/slide limits')
def enable_tool_bar(self, item): def enable_tool_bar(self, item):
""" """
@ -737,7 +736,7 @@ class SlideController(QtWidgets.QWidget, LogMixin, RegistryProperties):
self.play_slides_loop.setIcon(UiIcons().clock) self.play_slides_loop.setIcon(UiIcons().clock)
self.play_slides_loop.setText(UiStrings().PlaySlidesInLoop) self.play_slides_loop.setText(UiStrings().PlaySlidesInLoop)
if item.is_text(): if item.is_text():
if (Settings().value(self.main_window.songs_settings_section + '/display songbar') and if (self.settings.value(self.main_window.songs_settings_section + '/display songbar') and
not self.song_menu.menu().isEmpty()): not self.song_menu.menu().isEmpty()):
self.toolbar.set_widget_visible('song_menu', True) self.toolbar.set_widget_visible('song_menu', True)
if item.is_capable(ItemCapabilities.CanLoop) and len(item.slides) > 1: if item.is_capable(ItemCapabilities.CanLoop) and len(item.slides) > 1:
@ -1013,9 +1012,9 @@ class SlideController(QtWidgets.QWidget, LogMixin, RegistryProperties):
self.theme_screen.setChecked(False) self.theme_screen.setChecked(False)
self.desktop_screen.setChecked(False) self.desktop_screen.setChecked(False)
if checked: if checked:
Settings().setValue(self.main_window.general_settings_section + '/screen blank', 'blanked') self.settings.setValue(self.main_window.general_settings_section + '/screen blank', 'blanked')
else: else:
Settings().remove(self.main_window.general_settings_section + '/screen blank') self.settings.remove(self.main_window.general_settings_section + '/screen blank')
self.blank_plugin() self.blank_plugin()
self.update_preview() self.update_preview()
self.on_toggle_loop() self.on_toggle_loop()
@ -1034,9 +1033,9 @@ class SlideController(QtWidgets.QWidget, LogMixin, RegistryProperties):
self.theme_screen.setChecked(checked) self.theme_screen.setChecked(checked)
self.desktop_screen.setChecked(False) self.desktop_screen.setChecked(False)
if checked: if checked:
Settings().setValue(self.main_window.general_settings_section + '/screen blank', 'themed') self.settings.setValue(self.main_window.general_settings_section + '/screen blank', 'themed')
else: else:
Settings().remove(self.main_window.general_settings_section + '/screen blank') self.settings.remove(self.main_window.general_settings_section + '/screen blank')
self.blank_plugin() self.blank_plugin()
self.update_preview() self.update_preview()
self.on_toggle_loop() self.on_toggle_loop()
@ -1056,9 +1055,9 @@ class SlideController(QtWidgets.QWidget, LogMixin, RegistryProperties):
self.theme_screen.setChecked(False) self.theme_screen.setChecked(False)
self.desktop_screen.setChecked(checked) self.desktop_screen.setChecked(checked)
if checked: if checked:
Settings().setValue(self.main_window.general_settings_section + '/screen blank', 'hidden') self.settings.setValue(self.main_window.general_settings_section + '/screen blank', 'hidden')
else: else:
Settings().remove(self.main_window.general_settings_section + '/screen blank') self.settings.remove(self.main_window.general_settings_section + '/screen blank')
self.hide_plugin(checked) self.hide_plugin(checked)
self.update_preview() self.update_preview()
self.on_toggle_loop() self.on_toggle_loop()
@ -1145,7 +1144,7 @@ class SlideController(QtWidgets.QWidget, LogMixin, RegistryProperties):
return return
# If "click live slide to unblank" is enabled, unblank the display. And start = Item is sent to Live. # If "click live slide to unblank" is enabled, unblank the display. And start = Item is sent to Live.
# Note: If this if statement is placed at the bottom of this function instead of top slide transitions are lost. # Note: If this if statement is placed at the bottom of this function instead of top slide transitions are lost.
if self.is_live and Settings().value('core/click live slide to unblank'): if self.is_live and self.settings.value('core/click live slide to unblank'):
if not start: if not start:
Registry().execute('slidecontroller_live_unblank') Registry().execute('slidecontroller_live_unblank')
row = self.preview_widget.current_slide_number() row = self.preview_widget.current_slide_number()
@ -1353,7 +1352,7 @@ class SlideController(QtWidgets.QWidget, LogMixin, RegistryProperties):
self.play_slides_once.setText(UiStrings().PlaySlidesToEnd) self.play_slides_once.setText(UiStrings().PlaySlidesToEnd)
self.play_slides_menu.setDefaultAction(self.play_slides_loop) self.play_slides_menu.setDefaultAction(self.play_slides_loop)
self.play_slides_once.setChecked(False) self.play_slides_once.setChecked(False)
if Settings().value('core/click live slide to unblank'): if self.settings.value('core/click live slide to unblank'):
Registry().execute('slidecontroller_live_unblank') Registry().execute('slidecontroller_live_unblank')
else: else:
self.play_slides_loop.setIcon(UiIcons().clock) self.play_slides_loop.setIcon(UiIcons().clock)
@ -1378,7 +1377,7 @@ class SlideController(QtWidgets.QWidget, LogMixin, RegistryProperties):
self.play_slides_loop.setText(UiStrings().PlaySlidesInLoop) self.play_slides_loop.setText(UiStrings().PlaySlidesInLoop)
self.play_slides_menu.setDefaultAction(self.play_slides_once) self.play_slides_menu.setDefaultAction(self.play_slides_once)
self.play_slides_loop.setChecked(False) self.play_slides_loop.setChecked(False)
if Settings().value('core/click live slide to unblank'): if self.settings.value('core/click live slide to unblank'):
Registry().execute('slidecontroller_live_unblank') Registry().execute('slidecontroller_live_unblank')
else: else:
self.play_slides_once.setIcon(UiIcons().clock) self.play_slides_once.setIcon(UiIcons().clock)
@ -1423,7 +1422,7 @@ class SlideController(QtWidgets.QWidget, LogMixin, RegistryProperties):
Triggered when a preview slide item is double clicked Triggered when a preview slide item is double clicked
""" """
if self.service_item: if self.service_item:
if Settings().value('advanced/double click live') and Settings().value('core/auto unblank'): if self.settings.value('advanced/double click live') and self.settings.value('core/auto unblank'):
# Live and Preview have issues if we have video or presentations # Live and Preview have issues if we have video or presentations
# playing in both at the same time. # playing in both at the same time.
if self.service_item.is_command(): if self.service_item.is_command():

View File

@ -43,7 +43,7 @@ from openlp.core.lib.ui import critical_error_message_box
from openlp.core.widgets.edits import PathEdit from openlp.core.widgets.edits import PathEdit
from openlp.core.widgets.wizard import OpenLPWizard, WizardStrings from openlp.core.widgets.wizard import OpenLPWizard, WizardStrings
from openlp.plugins.bibles.lib.db import clean_filename from openlp.plugins.bibles.lib.db import clean_filename
from openlp.plugins.bibles.lib.importers.http import BGExtract, BSExtract, CWExtract from openlp.plugins.bibles.lib.importers.http import BGExtract, CWExtract
from openlp.plugins.bibles.lib.manager import BibleFormat from openlp.plugins.bibles.lib.manager import BibleFormat
@ -59,7 +59,8 @@ class WebDownload(object):
BibleGateway = 1 BibleGateway = 1
Bibleserver = 2 Bibleserver = 2
Names = ['Crosswalk', 'BibleGateway', 'Bibleserver'] # NOTE: BibleServer support has been disabled since we can't currently parse it. Re-add if/when fixed.
Names = ['Crosswalk', 'BibleGateway']
class BibleImportForm(OpenLPWizard): class BibleImportForm(OpenLPWizard):
@ -227,7 +228,8 @@ class BibleImportForm(OpenLPWizard):
self.web_bible_layout.setWidget(1, QtWidgets.QFormLayout.LabelRole, self.web_source_label) self.web_bible_layout.setWidget(1, QtWidgets.QFormLayout.LabelRole, self.web_source_label)
self.web_source_combo_box = QtWidgets.QComboBox(self.web_widget) self.web_source_combo_box = QtWidgets.QComboBox(self.web_widget)
self.web_source_combo_box.setObjectName('WebSourceComboBox') self.web_source_combo_box.setObjectName('WebSourceComboBox')
self.web_source_combo_box.addItems(['', '', '']) # NOTE: Set to 2 items since BibleServer has been disabled. Set to 3 if/when fixed
self.web_source_combo_box.addItems(['', ''])
self.web_source_combo_box.setEnabled(False) self.web_source_combo_box.setEnabled(False)
self.web_bible_layout.setWidget(1, QtWidgets.QFormLayout.FieldRole, self.web_source_combo_box) self.web_bible_layout.setWidget(1, QtWidgets.QFormLayout.FieldRole, self.web_source_combo_box)
self.web_translation_label = QtWidgets.QLabel(self.web_bible_tab) self.web_translation_label = QtWidgets.QLabel(self.web_bible_tab)
@ -239,7 +241,8 @@ class BibleImportForm(OpenLPWizard):
self.web_translation_combo_box.setEnabled(False) self.web_translation_combo_box.setEnabled(False)
self.web_bible_layout.setWidget(2, QtWidgets.QFormLayout.FieldRole, self.web_translation_combo_box) self.web_bible_layout.setWidget(2, QtWidgets.QFormLayout.FieldRole, self.web_translation_combo_box)
self.web_progress_bar = QtWidgets.QProgressBar(self) self.web_progress_bar = QtWidgets.QProgressBar(self)
self.web_progress_bar.setRange(0, 3) # NOTE: Set to 2 since BibleServer has been disabled. Set to 3 if/when fixed
self.web_progress_bar.setRange(0, 2)
self.web_progress_bar.setObjectName('WebTranslationProgressBar') self.web_progress_bar.setObjectName('WebTranslationProgressBar')
self.web_progress_bar.setVisible(False) self.web_progress_bar.setVisible(False)
self.web_bible_layout.setWidget(3, QtWidgets.QFormLayout.SpanningRole, self.web_progress_bar) self.web_bible_layout.setWidget(3, QtWidgets.QFormLayout.SpanningRole, self.web_progress_bar)
@ -400,8 +403,9 @@ class BibleImportForm(OpenLPWizard):
'Crosswalk')) 'Crosswalk'))
self.web_source_combo_box.setItemText(WebDownload.BibleGateway, translate('BiblesPlugin.ImportWizardForm', self.web_source_combo_box.setItemText(WebDownload.BibleGateway, translate('BiblesPlugin.ImportWizardForm',
'BibleGateway')) 'BibleGateway'))
self.web_source_combo_box.setItemText(WebDownload.Bibleserver, translate('BiblesPlugin.ImportWizardForm', # NOTE: BibleServer support has been disabled since we can't currently parse it. Re-add if/when fixed.
'Bibleserver')) # self.web_source_combo_box.setItemText(WebDownload.Bibleserver, translate('BiblesPlugin.ImportWizardForm',
# 'Bibleserver'))
self.web_translation_label.setText(translate('BiblesPlugin.ImportWizardForm', 'Bible:')) self.web_translation_label.setText(translate('BiblesPlugin.ImportWizardForm', 'Bible:'))
self.sword_bible_label.setText(translate('BiblesPlugin.ImportWizardForm', 'Bibles:')) self.sword_bible_label.setText(translate('BiblesPlugin.ImportWizardForm', 'Bibles:'))
self.sword_folder_label.setText(translate('BiblesPlugin.ImportWizardForm', 'SWORD data folder:')) self.sword_folder_label.setText(translate('BiblesPlugin.ImportWizardForm', 'SWORD data folder:'))
@ -578,9 +582,9 @@ class BibleImportForm(OpenLPWizard):
self.web_progress_bar.setVisible(True) self.web_progress_bar.setVisible(True)
self.web_progress_bar.setValue(0) self.web_progress_bar.setValue(0)
# TODO: Where does critical_error_message_box get %s string from? # TODO: Where does critical_error_message_box get %s string from?
# NOTE: BibleServer support has been disabled since we can't currently parse it. Re-add if/when fixed.
for (download_type, extractor) in ((WebDownload.Crosswalk, CWExtract()), for (download_type, extractor) in ((WebDownload.Crosswalk, CWExtract()),
(WebDownload.BibleGateway, BGExtract()), (WebDownload.BibleGateway, BGExtract())):
(WebDownload.Bibleserver, BSExtract())):
try: try:
bibles = extractor.get_bibles_from_http() bibles = extractor.get_bibles_from_http()
except (urllib.error.URLError, ConnectionError): except (urllib.error.URLError, ConnectionError):

View File

@ -310,7 +310,7 @@ class PresentationMediaItem(MediaManagerItem):
image_path = doc.get_temp_folder() / 'mainslide{number:0>3d}.png'.format(number=i) image_path = doc.get_temp_folder() / 'mainslide{number:0>3d}.png'.format(number=i)
thumbnail_path = doc.get_thumbnail_folder() / 'slide{number:d}.png'.format(number=i) thumbnail_path = doc.get_thumbnail_folder() / 'slide{number:d}.png'.format(number=i)
while image_path.is_file(): while image_path.is_file():
service_item.add_from_image(str(image_path), file_name, thumbnail=str(thumbnail_path)) service_item.add_from_image(image_path, file_name, thumbnail=str(thumbnail_path))
i += 1 i += 1
image_path = doc.get_temp_folder() / 'mainslide{number:0>3d}.png'.format(number=i) image_path = doc.get_temp_folder() / 'mainslide{number:0>3d}.png'.format(number=i)
thumbnail_path = doc.get_thumbnail_folder() / 'slide{number:d}.png'.format(number=i) thumbnail_path = doc.get_thumbnail_folder() / 'slide{number:d}.png'.format(number=i)

View File

@ -56,6 +56,7 @@ WIN32_MODULES = [
LINUX_MODULES = [ LINUX_MODULES = [
# Optical drive detection. # Optical drive detection.
'dbus', 'dbus',
'distro',
'Xlib', 'Xlib',
] ]

View File

@ -101,6 +101,7 @@ using a computer and a data projector.""",
'beautifulsoup4', 'beautifulsoup4',
'chardet', 'chardet',
'dbus-python; platform_system=="Linux"', 'dbus-python; platform_system=="Linux"',
'distro; platform_system=="Linux"',
'lxml', 'lxml',
'Mako', 'Mako',
'pymediainfo >= 2.2', 'pymediainfo >= 2.2',

View File

@ -23,6 +23,7 @@ All the tests
""" """
import os import os
from tempfile import mkstemp from tempfile import mkstemp
from unittest.mock import MagicMock
import pytest import pytest
from PyQt5 import QtCore, QtWidgets from PyQt5 import QtCore, QtWidgets
@ -39,24 +40,34 @@ def qapp():
del app del app
@pytest.fixture
def registry():
"""An instance of the Registry"""
Registry.create()
@pytest.yield_fixture @pytest.yield_fixture
def settings(qapp): def settings(qapp, registry):
"""A Settings() instance""" """A Settings() instance"""
fd, ini_file = mkstemp('.ini') fd, ini_file = mkstemp('.ini')
Settings.set_filename(ini_file) Settings.set_filename(ini_file)
Registry.create()
Settings().setDefaultFormat(QtCore.QSettings.IniFormat) Settings().setDefaultFormat(QtCore.QSettings.IniFormat)
# Needed on windows to make sure a Settings object is available during the tests # Needed on windows to make sure a Settings object is available during the tests
sets = Settings() sets = Settings()
sets.setValue('themes/global theme', 'my_theme') sets.setValue('themes/global theme', 'my_theme')
Registry().register('settings', set) Registry().register('settings', sets)
yield sets yield sets
del sets del sets
os.close(fd) os.close(fd)
os.unlink(Settings().fileName()) os.unlink(Settings().fileName())
@pytest.fixture @pytest.yield_fixture
def registry(): def mock_settings(registry):
"""An instance of the Registry""" """A Mock Settings() instance"""
Registry.create() # Create and register a mock settings object to work with
mock_settings = MagicMock()
Registry().register('settings', mock_settings)
yield mock_settings
Registry().remove('settings')
del mock_settings

View File

@ -22,11 +22,11 @@
Functional tests to test the AppLocation class and related methods. Functional tests to test the AppLocation class and related methods.
""" """
from pathlib import Path from pathlib import Path
from unittest import TestCase from unittest import TestCase, skipUnless
from unittest.mock import MagicMock, call, patch from unittest.mock import MagicMock, call, patch
from openlp.core.common import Singleton, clean_button_text, de_hump, extension_loader, is_linux, is_macosx, is_win, \ from openlp.core.common import Singleton, clean_button_text, de_hump, extension_loader, is_linux, is_macosx, is_win, \
normalize_str, path_to_module, trace_error_handler is_64bit_instance, normalize_str, path_to_module, trace_error_handler
class TestCommonFunctions(TestCase): class TestCommonFunctions(TestCase):
@ -243,7 +243,7 @@ class TestCommonFunctions(TestCase):
# GIVEN: Mocked out objects # GIVEN: Mocked out objects
with patch('openlp.core.common.os') as mocked_os, patch('openlp.core.common.sys') as mocked_sys: with patch('openlp.core.common.os') as mocked_os, patch('openlp.core.common.sys') as mocked_sys:
# WHEN: The mocked os.name and sys.platform are set to 'posix' and 'linux3' repectivly # WHEN: The mocked os.name and sys.platform are set to 'posix' and 'linux3' repectively
mocked_os.name = 'posix' mocked_os.name = 'posix'
mocked_sys.platform = 'linux3' mocked_sys.platform = 'linux3'
@ -252,6 +252,40 @@ class TestCommonFunctions(TestCase):
assert is_win() is False, 'is_win() should return False' assert is_win() is False, 'is_win() should return False'
assert is_macosx() is False, 'is_macosx() should return False' assert is_macosx() is False, 'is_macosx() should return False'
@skipUnless(is_linux(), 'This can only run on Linux')
def test_is_linux_distro(self):
"""
Test the is_linux() function for a particular Linux distribution
"""
# GIVEN: Mocked out objects
with patch('openlp.core.common.os') as mocked_os, \
patch('openlp.core.common.sys') as mocked_sys, \
patch('openlp.core.common.distro_id') as mocked_distro_id:
# WHEN: The mocked os.name and sys.platform are set to 'posix' and 'linux3' repectively
# and the distro is Fedora
mocked_os.name = 'posix'
mocked_sys.platform = 'linux3'
mocked_distro_id.return_value = 'fedora'
# THEN: The three platform functions should perform properly
assert is_linux(distro='fedora') is True, 'is_linux(distro="fedora") should return True'
assert is_win() is False, 'is_win() should return False'
assert is_macosx() is False, 'is_macosx() should return False'
def test_is_64bit_instance(self):
"""
Test the is_64bit_instance() function
"""
# GIVEN: Mocked out objects
with patch('openlp.core.common.sys') as mocked_sys:
# WHEN: The mocked sys.maxsize is set to 32-bit
mocked_sys.maxsize = 2**32
# THEN: The result should be False
assert is_64bit_instance() is False, 'is_64bit_instance() should return False'
def test_normalize_str_leaves_newlines(self): def test_normalize_str_leaves_newlines(self):
# GIVEN: a string containing newlines # GIVEN: a string containing newlines
string = 'something\nelse' string = 'something\nelse'

View File

@ -25,6 +25,7 @@ from unittest import TestCase
from unittest.mock import MagicMock, patch from unittest.mock import MagicMock, patch
from openlp.core.common.registry import Registry from openlp.core.common.registry import Registry
from openlp.core.ui import DisplayControllerType
from openlp.core.ui.media.mediacontroller import MediaController from openlp.core.ui.media.mediacontroller import MediaController
from openlp.core.ui.media import ItemMediaInfo from openlp.core.ui.media import ItemMediaInfo
from tests.helpers.testmixin import TestMixin from tests.helpers.testmixin import TestMixin
@ -202,3 +203,122 @@ class TestMediaController(TestCase, TestMixin):
# THEN you can determine the run time # THEN you can determine the run time
assert results == test_data[1], 'The correct duration is returned for ' + test_data[0] assert results == test_data[1], 'The correct duration is returned for ' + test_data[0]
def test_on_media_play(self):
"""
Test the on_media_play method
"""
# GIVEN: A mocked live controller and a mocked media_play() method
mocked_live_controller = MagicMock()
Registry().register('live_controller', mocked_live_controller)
media_controller = MediaController()
media_controller.media_play = MagicMock()
# WHEN: the on_media_play() method is called
media_controller.on_media_play()
# The mocked live controller should be called
media_controller.media_play.assert_called_once_with(mocked_live_controller, False)
def test_on_media_pause(self):
"""
Test the on_media_pause method
"""
# GIVEN: A mocked live controller and a mocked media_pause() method
mocked_live_controller = MagicMock()
Registry().register('live_controller', mocked_live_controller)
media_controller = MediaController()
media_controller.media_pause = MagicMock()
# WHEN: the on_media_pause() method is called
media_controller.on_media_pause()
# The mocked live controller should be called
media_controller.media_pause.assert_called_once_with(mocked_live_controller)
def test_on_media_stop(self):
"""
Test the on_media_stop method
"""
# GIVEN: A mocked live controller and a mocked media_stop() method
mocked_live_controller = MagicMock()
Registry().register('live_controller', mocked_live_controller)
media_controller = MediaController()
media_controller.media_stop = MagicMock()
# WHEN: the on_media_stop() method is called
media_controller.on_media_stop()
# The mocked live controller should be called
media_controller.media_stop.assert_called_once_with(mocked_live_controller)
def test_display_controllers_live(self):
"""
Test that the display_controllers() method returns the live controller when requested
"""
# GIVEN: A mocked live controller
media_controller = MediaController()
mocked_live_controller = MagicMock()
mocked_preview_controller = MagicMock()
Registry().register('live_controller', mocked_live_controller)
Registry().register('preview_controller', mocked_preview_controller)
# WHEN: display_controllers() is called with DisplayControllerType.Live
controller = media_controller.display_controllers(DisplayControllerType.Live)
# THEN: the controller should be the live controller
assert controller is mocked_live_controller
def test_display_controllers_preview(self):
"""
Test that the display_controllers() method returns the preview controller when requested
"""
# GIVEN: A mocked live controller
media_controller = MediaController()
mocked_live_controller = MagicMock()
mocked_preview_controller = MagicMock()
Registry().register('live_controller', mocked_live_controller)
Registry().register('preview_controller', mocked_preview_controller)
# WHEN: display_controllers() is called with DisplayControllerType.Preview
controller = media_controller.display_controllers(DisplayControllerType.Preview)
# THEN: the controller should be the live controller
assert controller is mocked_preview_controller
def test_set_controls_visible(self):
"""
Test that "set_controls_visible" sets the media controls on the controller to be visible or not
"""
# GIVEN: A mocked controller
mocked_controller = MagicMock()
# WHEN: Set to visible
MediaController.set_controls_visible(mocked_controller, True)
# THEN: The media controls should have been set to visible
mocked_controller.mediabar.setVisible.assert_called_once_with(True)
@patch('openlp.core.ui.media.mediacontroller.ItemMediaInfo')
def test_setup_display(self, MockItemMediaInfo):
"""
Test that the display/controllers are set up correctly
"""
# GIVEN: A media controller object and some mocks
mocked_media_info = MagicMock()
MockItemMediaInfo.return_value = mocked_media_info
media_controller = MediaController()
media_controller.vlc_player = MagicMock()
mocked_display = MagicMock()
media_controller._define_display = MagicMock(return_value=mocked_display)
media_controller.vlc_player = MagicMock()
controller = MagicMock()
# WHEN: setup_display() is called
media_controller.setup_display(controller, True)
# THEN: The right calls should have been made
assert controller.media_info == mocked_media_info
assert controller.has_audio is False
media_controller._define_display.assert_called_once_with(controller)
media_controller.vlc_player.setup(controller, mocked_display, False)

View File

@ -21,7 +21,6 @@
""" """
Package to test the openlp.core.ui.slidecontroller package. Package to test the openlp.core.ui.slidecontroller package.
""" """
from unittest import TestCase
from unittest.mock import MagicMock, patch from unittest.mock import MagicMock, patch
from PyQt5 import QtCore, QtGui from PyQt5 import QtCore, QtGui
@ -32,15 +31,7 @@ from openlp.core.ui.slidecontroller import NON_TEXT_MENU, WIDE_MENU, InfoLabel,
SlideController SlideController
class TestSlideController(TestCase): def test_initial_slide_controller(registry):
def setUp(self):
"""
Set up the components need for all tests.
"""
Registry.create()
def test_initial_slide_controller(self):
""" """
Test the initial slide controller state . Test the initial slide controller state .
""" """
@ -51,7 +42,8 @@ class TestSlideController(TestCase):
# THEN: The controller should not be a live controller. # THEN: The controller should not be a live controller.
assert slide_controller.is_live is False, 'The base slide controller should not be a live controller' assert slide_controller.is_live is False, 'The base slide controller should not be a live controller'
def test_text_service_item_blank(self):
def test_text_service_item_blank():
""" """
Test that loading a text-based service item into the slide controller sets the correct blank menu Test that loading a text-based service item into the slide controller sets the correct blank menu
""" """
@ -70,7 +62,8 @@ class TestSlideController(TestCase):
# THEN: the call to set the visible items on the toolbar should be correct # THEN: the call to set the visible items on the toolbar should be correct
toolbar.set_widget_visible.assert_called_with(WIDE_MENU, True) toolbar.set_widget_visible.assert_called_with(WIDE_MENU, True)
def test_non_text_service_item_blank(self):
def test_non_text_service_item_blank():
""" """
Test that loading a non-text service item into the slide controller sets the correct blank menu Test that loading a non-text service item into the slide controller sets the correct blank menu
""" """
@ -89,14 +82,13 @@ class TestSlideController(TestCase):
# THEN: then call set up the toolbar to blank the display screen. # THEN: then call set up the toolbar to blank the display screen.
toolbar.set_widget_visible.assert_called_with(NON_TEXT_MENU, True) toolbar.set_widget_visible.assert_called_with(NON_TEXT_MENU, True)
@patch('openlp.core.ui.slidecontroller.Settings')
def test_receive_spin_delay(self, MockedSettings): def test_receive_spin_delay(mock_settings):
""" """
Test that the spin box is updated accordingly after a call to receive_spin_delay() Test that the spin box is updated accordingly after a call to receive_spin_delay()
""" """
# GIVEN: A new SlideController instance. # GIVEN: A new SlideController instance.
mocked_value = MagicMock(return_value=1) mock_settings.value.return_value = 1
MockedSettings.return_value = MagicMock(value=mocked_value)
mocked_delay_spin_box = MagicMock() mocked_delay_spin_box = MagicMock()
slide_controller = SlideController(None) slide_controller = SlideController(None)
slide_controller.delay_spin_box = mocked_delay_spin_box slide_controller.delay_spin_box = mocked_delay_spin_box
@ -104,11 +96,13 @@ class TestSlideController(TestCase):
# WHEN: The receive_spin_delay() method is called # WHEN: The receive_spin_delay() method is called
slide_controller.receive_spin_delay() slide_controller.receive_spin_delay()
# THEN: The Settings()value() and delay_spin_box.setValue() methods should have been called correctly # THEN: The settings value() and delay_spin_box.setValue() methods should have been called correctly
mocked_value.assert_called_with('core/loop delay') msv = mock_settings.value
msv.assert_called_with('core/loop delay')
mocked_delay_spin_box.setValue.assert_called_with(1) mocked_delay_spin_box.setValue.assert_called_with(1)
def test_toggle_display_blank(self):
def test_toggle_display_blank():
""" """
Check that the toggle_display('blank') method calls the on_blank_display() method Check that the toggle_display('blank') method calls the on_blank_display() method
""" """
@ -129,7 +123,8 @@ class TestSlideController(TestCase):
assert 0 == mocked_on_theme_display.call_count, 'on_theme_display should not have been called' assert 0 == mocked_on_theme_display.call_count, 'on_theme_display should not have been called'
assert 0 == mocked_on_hide_display.call_count, 'on_hide_display should not have been called' assert 0 == mocked_on_hide_display.call_count, 'on_hide_display should not have been called'
def test_toggle_display_hide(self):
def test_toggle_display_hide():
""" """
Check that the toggle_display('hide') method calls the on_blank_display() method Check that the toggle_display('hide') method calls the on_blank_display() method
""" """
@ -150,7 +145,8 @@ class TestSlideController(TestCase):
assert 0 == mocked_on_theme_display.call_count, 'on_theme_display should not have been called' assert 0 == mocked_on_theme_display.call_count, 'on_theme_display should not have been called'
assert 0 == mocked_on_hide_display.call_count, 'on_hide_display should not have been called' assert 0 == mocked_on_hide_display.call_count, 'on_hide_display should not have been called'
def test_toggle_display_theme(self):
def test_toggle_display_theme():
""" """
Check that the toggle_display('theme') method calls the on_theme_display() method Check that the toggle_display('theme') method calls the on_theme_display() method
""" """
@ -171,7 +167,8 @@ class TestSlideController(TestCase):
assert 0 == mocked_on_blank_display.call_count, 'on_blank_display should not have been called' assert 0 == mocked_on_blank_display.call_count, 'on_blank_display should not have been called'
assert 0 == mocked_on_hide_display.call_count, 'on_hide_display should not have been called' assert 0 == mocked_on_hide_display.call_count, 'on_hide_display should not have been called'
def test_toggle_display_desktop(self):
def test_toggle_display_desktop():
""" """
Check that the toggle_display('desktop') method calls the on_hide_display() method Check that the toggle_display('desktop') method calls the on_hide_display() method
""" """
@ -192,7 +189,8 @@ class TestSlideController(TestCase):
assert 0 == mocked_on_blank_display.call_count, 'on_blank_display should not have been called' assert 0 == mocked_on_blank_display.call_count, 'on_blank_display should not have been called'
assert 0 == mocked_on_theme_display.call_count, 'on_theme_display should not have been called' assert 0 == mocked_on_theme_display.call_count, 'on_theme_display should not have been called'
def test_toggle_display_show(self):
def test_toggle_display_show():
""" """
Check that the toggle_display('show') method calls all the on_X_display() methods Check that the toggle_display('show') method calls all the on_X_display() methods
""" """
@ -213,7 +211,8 @@ class TestSlideController(TestCase):
mocked_on_theme_display.assert_called_once_with(False) mocked_on_theme_display.assert_called_once_with(False)
mocked_on_hide_display.assert_called_once_with(False) mocked_on_hide_display.assert_called_once_with(False)
def test_on_go_live_preview_controller(self):
def test_on_go_live_preview_controller():
""" """
Test that when the on_go_preview() method is called the message is sent to the preview controller and focus is Test that when the on_go_preview() method is called the message is sent to the preview controller and focus is
set correctly. set correctly.
@ -240,7 +239,8 @@ class TestSlideController(TestCase):
# THEN: the preview controller should have the service item and the focus set to live # THEN: the preview controller should have the service item and the focus set to live
mocked_preview_controller.preview_widget.setFocus.assert_called_once_with() mocked_preview_controller.preview_widget.setFocus.assert_called_once_with()
def test_on_go_live_live_controller(self):
def test_on_go_live_live_controller():
""" """
Test that when the on_go_live() method is called the message is sent to the live controller and focus is Test that when the on_go_live() method is called the message is sent to the live controller and focus is
set correctly. set correctly.
@ -268,7 +268,8 @@ class TestSlideController(TestCase):
mocked_live_controller.add_service_manager_item.assert_called_once_with(mocked_service_item, 1) mocked_live_controller.add_service_manager_item.assert_called_once_with(mocked_service_item, 1)
mocked_live_controller.preview_widget.setFocus.assert_called_once_with() mocked_live_controller.preview_widget.setFocus.assert_called_once_with()
def test_on_go_live_service_manager(self):
def test_on_go_live_service_manager():
""" """
Test that when the on_go_live() method is called the message is sent to the live controller and focus is Test that when the on_go_live() method is called the message is sent to the live controller and focus is
set correctly. set correctly.
@ -299,7 +300,8 @@ class TestSlideController(TestCase):
mocked_service_manager.preview_live.assert_called_once_with(42, 1) mocked_service_manager.preview_live.assert_called_once_with(42, 1)
mocked_live_controller.preview_widget.setFocus.assert_called_once_with() mocked_live_controller.preview_widget.setFocus.assert_called_once_with()
def test_service_previous(self):
def test_service_previous():
""" """
Check that calling the service_previous() method adds the previous key to the queue and processes the queue Check that calling the service_previous() method adds the previous key to the queue and processes the queue
""" """
@ -317,7 +319,8 @@ class TestSlideController(TestCase):
mocked_keypress_queue.append.assert_called_once_with(ServiceItemAction.Previous) mocked_keypress_queue.append.assert_called_once_with(ServiceItemAction.Previous)
mocked_process_queue.assert_called_once_with() mocked_process_queue.assert_called_once_with()
def test_service_next(self):
def test_service_next():
""" """
Check that calling the service_next() method adds the next key to the queue and processes the queue Check that calling the service_next() method adds the next key to the queue and processes the queue
""" """
@ -335,16 +338,14 @@ class TestSlideController(TestCase):
mocked_keypress_queue.append.assert_called_once_with(ServiceItemAction.Next) mocked_keypress_queue.append.assert_called_once_with(ServiceItemAction.Next)
mocked_process_queue.assert_called_once_with() mocked_process_queue.assert_called_once_with()
@patch('openlp.core.ui.slidecontroller.Settings')
def test_update_slide_limits(self, MockedSettings): def test_update_slide_limits(mock_settings):
""" """
Test that calling the update_slide_limits() method updates the slide limits Test that calling the update_slide_limits() method updates the slide limits
""" """
# GIVEN: A mocked out Settings object, a new SlideController and a mocked out main_window # GIVEN: A mocked out Settings object, a new SlideController and a mocked out main_window
mocked_value = MagicMock(return_value=10) mock_settings.value.return_value = 10
MockedSettings.return_value = MagicMock(value=mocked_value)
mocked_main_window = MagicMock(advanced_settings_section='advanced') mocked_main_window = MagicMock(advanced_settings_section='advanced')
Registry.create()
Registry().register('main_window', mocked_main_window) Registry().register('main_window', mocked_main_window)
slide_controller = SlideController(None) slide_controller = SlideController(None)
@ -352,10 +353,12 @@ class TestSlideController(TestCase):
slide_controller.update_slide_limits() slide_controller.update_slide_limits()
# THEN: The value of slide_limits should be 10 # THEN: The value of slide_limits should be 10
mocked_value.assert_called_once_with('advanced/slide limits') msv = mock_settings.value
msv.assert_called_once_with('advanced/slide limits')
assert 10 == slide_controller.slide_limits, 'Slide limits should have been updated to 10' assert 10 == slide_controller.slide_limits, 'Slide limits should have been updated to 10'
def test_enable_tool_bar_live(self):
def test_enable_tool_bar_live():
""" """
Check that when enable_tool_bar on a live slide controller is called, enable_live_tool_bar is called Check that when enable_tool_bar on a live slide controller is called, enable_live_tool_bar is called
""" """
@ -375,7 +378,8 @@ class TestSlideController(TestCase):
mocked_enable_live_tool_bar.assert_called_once_with(mocked_service_item) mocked_enable_live_tool_bar.assert_called_once_with(mocked_service_item)
assert 0 == mocked_enable_preview_tool_bar.call_count, 'The preview method should not have been called' assert 0 == mocked_enable_preview_tool_bar.call_count, 'The preview method should not have been called'
def test_enable_tool_bar_preview(self):
def test_enable_tool_bar_preview():
""" """
Check that when enable_tool_bar on a preview slide controller is called, enable_preview_tool_bar is called Check that when enable_tool_bar on a preview slide controller is called, enable_preview_tool_bar is called
""" """
@ -395,7 +399,8 @@ class TestSlideController(TestCase):
mocked_enable_preview_tool_bar.assert_called_once_with(mocked_service_item) mocked_enable_preview_tool_bar.assert_called_once_with(mocked_service_item)
assert 0 == mocked_enable_live_tool_bar.call_count, 'The live method should not have been called' assert 0 == mocked_enable_live_tool_bar.call_count, 'The live method should not have been called'
def test_refresh_service_item_text(self):
def test_refresh_service_item_text():
""" """
Test that the refresh_service_item() method refreshes a text service item Test that the refresh_service_item() method refreshes a text service item
""" """
@ -418,7 +423,8 @@ class TestSlideController(TestCase):
mocked_service_item.render.assert_called_once_with() mocked_service_item.render.assert_called_once_with()
mocked_process_item.assert_called_once_with(mocked_service_item, 5) mocked_process_item.assert_called_once_with(mocked_service_item, 5)
def test_refresh_service_item_image(self):
def test_refresh_service_item_image():
""" """
Test that the refresh_service_item() method refreshes a image service item Test that the refresh_service_item() method refreshes a image service item
""" """
@ -441,7 +447,8 @@ class TestSlideController(TestCase):
mocked_service_item.render.assert_called_once_with() mocked_service_item.render.assert_called_once_with()
mocked_process_item.assert_called_once_with(mocked_service_item, 5) mocked_process_item.assert_called_once_with(mocked_service_item, 5)
def test_refresh_service_item_not_image_or_text(self):
def test_refresh_service_item_not_image_or_text():
""" """
Test that the refresh_service_item() method does not refresh a service item if it's neither text or an image Test that the refresh_service_item() method does not refresh a service item if it's neither text or an image
""" """
@ -464,7 +471,8 @@ class TestSlideController(TestCase):
assert 0 == mocked_service_item.render.call_count, 'The render() method should not have been called' assert 0 == mocked_service_item.render.call_count, 'The render() method should not have been called'
assert 0 == mocked_process_item.call_count, 'The mocked_process_item() method should not have been called' assert 0 == mocked_process_item.call_count, 'The mocked_process_item() method should not have been called'
def test_add_service_item_with_song_edit(self):
def test_add_service_item_with_song_edit():
""" """
Test the add_service_item() method when song_edit is True Test the add_service_item() method when song_edit is True
""" """
@ -483,7 +491,8 @@ class TestSlideController(TestCase):
assert slide_controller.song_edit is False, 'song_edit should be False' assert slide_controller.song_edit is False, 'song_edit should be False'
mocked_process_item.assert_called_once_with(mocked_item, 2) mocked_process_item.assert_called_once_with(mocked_item, 2)
def test_add_service_item_without_song_edit(self):
def test_add_service_item_without_song_edit():
""" """
Test the add_service_item() method when song_edit is False Test the add_service_item() method when song_edit is False
""" """
@ -502,7 +511,8 @@ class TestSlideController(TestCase):
assert slide_controller.song_edit is False, 'song_edit should be False' assert slide_controller.song_edit is False, 'song_edit should be False'
mocked_process_item.assert_called_once_with(mocked_item, 0) mocked_process_item.assert_called_once_with(mocked_item, 0)
def test_replace_service_manager_item_different_items(self):
def test_replace_service_manager_item_different_items():
""" """
Test that when the service items are not the same, nothing happens Test that when the service items are not the same, nothing happens
""" """
@ -523,7 +533,8 @@ class TestSlideController(TestCase):
assert 0 == mocked_preview_widget.current_slide_number.call_count, \ assert 0 == mocked_preview_widget.current_slide_number.call_count, \
'The preview_widget current_slide_number.() method should not have been called' 'The preview_widget current_slide_number.() method should not have been called'
def test_replace_service_manager_item_same_item(self):
def test_replace_service_manager_item_same_item():
""" """
Test that when the service item is the same, the service item is reprocessed Test that when the service item is the same, the service item is reprocessed
""" """
@ -544,7 +555,8 @@ class TestSlideController(TestCase):
mocked_preview_widget.current_slide_number.assert_called_with() mocked_preview_widget.current_slide_number.assert_called_with()
mocked_process_item.assert_called_once_with(mocked_item, 7) mocked_process_item.assert_called_once_with(mocked_item, 7)
def test_on_slide_blank(self):
def test_on_slide_blank():
""" """
Test on_slide_blank Test on_slide_blank
""" """
@ -558,7 +570,8 @@ class TestSlideController(TestCase):
# THEN: on_blank_display should have been called with True # THEN: on_blank_display should have been called with True
slide_controller.on_blank_display.assert_called_once_with(True) slide_controller.on_blank_display.assert_called_once_with(True)
def test_on_slide_unblank(self):
def test_on_slide_unblank():
""" """
Test on_slide_unblank Test on_slide_unblank
""" """
@ -572,7 +585,8 @@ class TestSlideController(TestCase):
# THEN: on_blank_display should have been called with False # THEN: on_blank_display should have been called with False
slide_controller.on_blank_display.assert_called_once_with(False) slide_controller.on_blank_display.assert_called_once_with(False)
def test_on_slide_selected_index_no_service_item(self):
def test_on_slide_selected_index_no_service_item():
""" """
Test that when there is no service item, the on_slide_selected_index() method returns immediately Test that when there is no service item, the on_slide_selected_index() method returns immediately
""" """
@ -587,8 +601,9 @@ class TestSlideController(TestCase):
# THEN: It should have exited early # THEN: It should have exited early
assert 0 == mocked_item.is_command.call_count, 'The service item should have not been called' assert 0 == mocked_item.is_command.call_count, 'The service item should have not been called'
@patch.object(Registry, 'execute')
def test_on_slide_selected_index_service_item_command(self, mocked_execute): @patch.object(Registry, 'execute')
def test_on_slide_selected_index_service_item_command(mocked_execute):
""" """
Test that when there is a command service item, the command is executed Test that when there is a command service item, the command is executed
""" """
@ -617,8 +632,9 @@ class TestSlideController(TestCase):
assert 0 == mocked_preview_widget.change_slide.call_count, 'Change slide should not have been called' assert 0 == mocked_preview_widget.change_slide.call_count, 'Change slide should not have been called'
assert 0 == mocked_slide_selected.call_count, 'slide_selected should not have been called' assert 0 == mocked_slide_selected.call_count, 'slide_selected should not have been called'
@patch.object(Registry, 'execute')
def test_on_slide_selected_index_service_item_not_command(self, mocked_execute): @patch.object(Registry, 'execute')
def test_on_slide_selected_index_service_item_not_command(mocked_execute):
""" """
Test that when there is a service item but it's not a command, the preview widget is updated Test that when there is a service item but it's not a command, the preview widget is updated
""" """
@ -646,8 +662,9 @@ class TestSlideController(TestCase):
mocked_preview_widget.change_slide.assert_called_once_with(7) mocked_preview_widget.change_slide.assert_called_once_with(7)
mocked_slide_selected.assert_called_once_with() mocked_slide_selected.assert_called_once_with()
@patch.object(Registry, 'execute')
def test_process_item(self, mocked_execute): @patch.object(Registry, 'execute')
def test_process_item(mocked_execute):
""" """
Test that presentation service-items is closed when followed by a media service-item Test that presentation service-items is closed when followed by a media service-item
""" """
@ -667,8 +684,8 @@ class TestSlideController(TestCase):
mocked_media_item.is_image.return_value = False mocked_media_item.is_image.return_value = False
mocked_media_item.from_service = False mocked_media_item.from_service = False
mocked_media_item.get_frames.return_value = [] mocked_media_item.get_frames.return_value = []
Registry.create()
mocked_main_window = MagicMock() mocked_main_window = MagicMock()
Registry.create()
Registry().register('main_window', mocked_main_window) Registry().register('main_window', mocked_main_window)
Registry().register('media_controller', MagicMock()) Registry().register('media_controller', MagicMock())
slide_controller = SlideController(None) slide_controller = SlideController(None)
@ -693,7 +710,8 @@ class TestSlideController(TestCase):
assert 'mocked_presentation_item_stop' == mocked_execute.call_args_list[1][0][0], \ assert 'mocked_presentation_item_stop' == mocked_execute.call_args_list[1][0][0], \
'The presentation should have been stopped.' 'The presentation should have been stopped.'
def test_live_stolen_focus_shortcuts(self):
def test_live_stolen_focus_shortcuts():
""" """
Test that all the needed shortcuts are available in scenarios where Live has stolen focus. Test that all the needed shortcuts are available in scenarios where Live has stolen focus.
These are found under def __add_actions_to_widget(self, widget): in slidecontroller.py These are found under def __add_actions_to_widget(self, widget): in slidecontroller.py
@ -721,20 +739,19 @@ class TestSlideController(TestCase):
slide_controller.theme_screen, slide_controller.blank_screen slide_controller.theme_screen, slide_controller.blank_screen
]) ])
@patch('openlp.core.ui.slidecontroller.Settings')
def test_on_preview_double_click_unblank_display(self, MockedSettings): def test_on_preview_double_click_unblank_display(mock_settings):
# GIVEN: A slide controller, actions needed, settins set to True. # GIVEN: A slide controller, actions needed, settings set to True.
slide_controller = SlideController(None) slide_controller = SlideController(None)
mocked_settings = MagicMock() mocked_settings = MagicMock()
mocked_settings.return_value = True mocked_settings.return_value = True
MockedSettings.return_value = mocked_settings mock_settings.return_value = mocked_settings
slide_controller.service_item = MagicMock() slide_controller.service_item = MagicMock()
slide_controller.service_item.is_media = MagicMock() slide_controller.service_item.is_media = MagicMock()
slide_controller.on_media_close = MagicMock() slide_controller.on_media_close = MagicMock()
slide_controller.on_go_live = MagicMock() slide_controller.on_go_live = MagicMock()
slide_controller.on_preview_add_to_service = MagicMock() slide_controller.on_preview_add_to_service = MagicMock()
slide_controller.media_reset = MagicMock() slide_controller.media_reset = MagicMock()
Registry.create()
Registry().set_flag('has doubleclick added item to service', True) Registry().set_flag('has doubleclick added item to service', True)
# WHEN: on_preview_double_click is called # WHEN: on_preview_double_click is called
@ -744,20 +761,17 @@ class TestSlideController(TestCase):
assert 1 == slide_controller.on_go_live.call_count, 'on_go_live should have been called once.' assert 1 == slide_controller.on_go_live.call_count, 'on_go_live should have been called once.'
assert 0 == slide_controller.on_preview_add_to_service.call_count, 'Should have not been called.' assert 0 == slide_controller.on_preview_add_to_service.call_count, 'Should have not been called.'
@patch('openlp.core.ui.slidecontroller.Settings')
def test_on_preview_double_click_add_to_service(self, MockedSettings): def test_on_preview_double_click_add_to_service(mock_settings):
# GIVEN: A slide controller, actions needed, settins set to False. # GIVEN: A slide controller, actions needed, settings set to False.
slide_controller = SlideController(None) slide_controller = SlideController(None)
mocked_settings = MagicMock() mock_settings.value.return_value = False
mocked_settings.value.return_value = False
MockedSettings.return_value = mocked_settings
slide_controller.service_item = MagicMock() slide_controller.service_item = MagicMock()
slide_controller.service_item.is_media = MagicMock() slide_controller.service_item.is_media = MagicMock()
slide_controller.on_media_close = MagicMock() slide_controller.on_media_close = MagicMock()
slide_controller.on_go_live = MagicMock() slide_controller.on_go_live = MagicMock()
slide_controller.on_preview_add_to_service = MagicMock() slide_controller.on_preview_add_to_service = MagicMock()
slide_controller.media_reset = MagicMock() slide_controller.media_reset = MagicMock()
Registry.create()
Registry().set_flag('has doubleclick added item to service', False) Registry().set_flag('has doubleclick added item to service', False)
# WHEN: on_preview_double_click is called # WHEN: on_preview_double_click is called
@ -767,9 +781,10 @@ class TestSlideController(TestCase):
assert 0 == slide_controller.on_go_live.call_count, 'on_go_live Should have not been called.' assert 0 == slide_controller.on_go_live.call_count, 'on_go_live Should have not been called.'
assert 1 == slide_controller.on_preview_add_to_service.call_count, 'Should have been called once.' assert 1 == slide_controller.on_preview_add_to_service.call_count, 'Should have been called once.'
@patch(u'openlp.core.ui.slidecontroller.SlideController.image_manager')
@patch(u'PyQt5.QtCore.QTimer.singleShot') @patch(u'openlp.core.ui.slidecontroller.SlideController.image_manager')
def test_update_preview_live(self, mocked_singleShot, mocked_image_manager): @patch(u'PyQt5.QtCore.QTimer.singleShot')
def test_update_preview_live(mocked_singleShot, mocked_image_manager):
""" """
Test that the preview screen is updated with a screen grab for live service items Test that the preview screen is updated with a screen grab for live service items
""" """
@ -809,9 +824,10 @@ class TestSlideController(TestCase):
assert 2 == mocked_singleShot.call_count, 'Timer to grab_maindisplay should have been called 2 times' assert 2 == mocked_singleShot.call_count, 'Timer to grab_maindisplay should have been called 2 times'
assert 0 == mocked_image_manager.get_image.call_count, 'image_manager not be called' assert 0 == mocked_image_manager.get_image.call_count, 'image_manager not be called'
@patch(u'openlp.core.ui.slidecontroller.SlideController.image_manager')
@patch(u'PyQt5.QtCore.QTimer.singleShot') @patch(u'openlp.core.ui.slidecontroller.SlideController.image_manager')
def test_update_preview_pres(self, mocked_singleShot, mocked_image_manager): @patch(u'PyQt5.QtCore.QTimer.singleShot')
def test_update_preview_pres(mocked_singleShot, mocked_image_manager):
""" """
Test that the preview screen is updated with the correct preview for presentation service items Test that the preview screen is updated with the correct preview for presentation service items
""" """
@ -849,11 +865,11 @@ class TestSlideController(TestCase):
# THEN: setPixmap and the image_manager should have been called # THEN: setPixmap and the image_manager should have been called
assert 1 == slide_controller.preview_display.set_single_image.call_count, 'set_single_image should be called' assert 1 == slide_controller.preview_display.set_single_image.call_count, 'set_single_image should be called'
assert 0 == mocked_singleShot.call_count, 'Timer to grab_maindisplay should not be called' assert 0 == mocked_singleShot.call_count, 'Timer to grab_maindisplay should not be called'
# assert 1 == mocked_image_manager.get_image.call_count, 'image_manager should be called'
@patch(u'openlp.core.ui.slidecontroller.SlideController.image_manager')
@patch(u'PyQt5.QtCore.QTimer.singleShot') @patch(u'openlp.core.ui.slidecontroller.SlideController.image_manager')
def test_update_preview_media(self, mocked_singleShot, mocked_image_manager): @patch(u'PyQt5.QtCore.QTimer.singleShot')
def test_update_preview_media(mocked_singleShot, mocked_image_manager):
""" """
Test that the preview screen is updated with the correct preview for media service items Test that the preview screen is updated with the correct preview for media service items
""" """
@ -893,9 +909,10 @@ class TestSlideController(TestCase):
assert 0 == mocked_singleShot.call_count, 'Timer to grab_maindisplay should not be called' assert 0 == mocked_singleShot.call_count, 'Timer to grab_maindisplay should not be called'
assert 0 == mocked_image_manager.get_image.call_count, 'image_manager should not be called' assert 0 == mocked_image_manager.get_image.call_count, 'image_manager should not be called'
@patch(u'openlp.core.ui.slidecontroller.SlideController.image_manager')
@patch(u'PyQt5.QtCore.QTimer.singleShot') @patch(u'openlp.core.ui.slidecontroller.SlideController.image_manager')
def test_update_preview_image(self, mocked_singleShot, mocked_image_manager): @patch(u'PyQt5.QtCore.QTimer.singleShot')
def test_update_preview_image(mocked_singleShot, mocked_image_manager):
""" """
Test that the preview screen is updated with the correct preview for image service items Test that the preview screen is updated with the correct preview for image service items
""" """
@ -935,14 +952,14 @@ class TestSlideController(TestCase):
assert 0 == mocked_image_manager.get_image.call_count, 'image_manager should not be called' assert 0 == mocked_image_manager.get_image.call_count, 'image_manager should not be called'
class TestInfoLabel(TestCase): def test_paint_event_text_fits():
def test_paint_event_text_fits(self):
""" """
Test the paintEvent method when text fits the label Test the paintEvent method when text fits the label
""" """
font = QtGui.QFont()
metrics = QtGui.QFontMetrics(font)
with patch('openlp.core.ui.slidecontroller.QtWidgets.QLabel'), \ with patch('openlp.core.ui.slidecontroller.QtWidgets.QLabel'), \
patch('openlp.core.ui.slidecontroller.QtGui.QFontMetrics') as MockFontMetrics, \
patch('openlp.core.ui.slidecontroller.QtGui.QPainter') as mocked_qpainter: patch('openlp.core.ui.slidecontroller.QtGui.QPainter') as mocked_qpainter:
# GIVEN: An instance of InfoLabel, with mocked text return, width and rect methods # GIVEN: An instance of InfoLabel, with mocked text return, width and rect methods
@ -955,28 +972,28 @@ class TestInfoLabel(TestCase):
info_label.rect = mocked_rect info_label.rect = mocked_rect
info_label.text = mocked_text info_label.text = mocked_text
info_label.width = mocked_width info_label.width = mocked_width
mocked_font_metrics = MagicMock()
mocked_font_metrics.elidedText.return_value = test_string
MockFontMetrics.return_value = mocked_font_metrics
# WHEN: The instance is wider than its text, and the paintEvent method is called # WHEN: The instance is wider than its text, and the paintEvent method is called
info_label.width.return_value = metrics.boundingRect(test_string).width() + 20
info_label.paintEvent(MagicMock()) info_label.paintEvent(MagicMock())
# THEN: The text should be drawn centered with the complete test_string # THEN: The text should be drawn centered with the complete test_string
mocked_qpainter().drawText.assert_called_once_with(mocked_rect(), QtCore.Qt.AlignCenter, test_string) mocked_qpainter().drawText.assert_called_once_with(mocked_rect(), QtCore.Qt.AlignCenter, test_string)
def test_paint_event_text_doesnt_fit(self):
def test_paint_event_text_doesnt_fit():
""" """
Test the paintEvent method when text fits the label Test the paintEvent method when text fits the label
""" """
font = QtGui.QFont()
metrics = QtGui.QFontMetrics(font)
with patch('openlp.core.ui.slidecontroller.QtWidgets.QLabel'), \ with patch('openlp.core.ui.slidecontroller.QtWidgets.QLabel'), \
patch('openlp.core.ui.slidecontroller.QtGui.QFontMetrics') as MockFontMetrics, \
patch('openlp.core.ui.slidecontroller.QtGui.QPainter') as mocked_qpainter: patch('openlp.core.ui.slidecontroller.QtGui.QPainter') as mocked_qpainter:
# GIVEN: An instance of InfoLabel, with mocked text return, width and rect methods # GIVEN: An instance of InfoLabel, with mocked text return, width and rect methods
info_label = InfoLabel() info_label = InfoLabel()
test_string = 'Label Text' test_string = 'Label Text'
elided_test_string = test_string[0:5] + '...'
mocked_rect = MagicMock() mocked_rect = MagicMock()
mocked_text = MagicMock() mocked_text = MagicMock()
mocked_width = MagicMock() mocked_width = MagicMock()
@ -984,18 +1001,19 @@ class TestInfoLabel(TestCase):
info_label.rect = mocked_rect info_label.rect = mocked_rect
info_label.text = mocked_text info_label.text = mocked_text
info_label.width = mocked_width info_label.width = mocked_width
mocked_font_metrics = MagicMock()
mocked_font_metrics.elidedText.return_value = elided_test_string
MockFontMetrics.return_value = mocked_font_metrics
# WHEN: The instance is narrower than its text, and the paintEvent method is called # WHEN: The instance is narrower than its text, and the paintEvent method is called
label_width = metrics.boundingRect(test_string).width() - 20
info_label.width.return_value = label_width
info_label.paintEvent(MagicMock()) info_label.paintEvent(MagicMock())
# THEN: The text should be drawn aligned left with an elided test_string # THEN: The text should be drawn aligned left with an elided test_string
elided_test_string = metrics.elidedText(test_string, QtCore.Qt.ElideRight, label_width)
mocked_qpainter().drawText.assert_called_once_with(mocked_rect(), QtCore.Qt.AlignLeft, elided_test_string) mocked_qpainter().drawText.assert_called_once_with(mocked_rect(), QtCore.Qt.AlignLeft, elided_test_string)
@patch('builtins.super')
def test_set_text(self, mocked_super): @patch('builtins.super')
def test_set_text(mocked_super):
""" """
Test the reimplemented setText method Test the reimplemented setText method
""" """
@ -1012,9 +1030,7 @@ class TestInfoLabel(TestCase):
mocked_super().setText.assert_called_once_with('Label Text') mocked_super().setText.assert_called_once_with('Label Text')
class TestLiveController(TestCase): def test_initial_live_controller():
def test_initial_live_controller(self):
""" """
Test the initial live slide controller state . Test the initial live slide controller state .
""" """
@ -1027,9 +1043,7 @@ class TestLiveController(TestCase):
assert live_controller.is_live is True, 'The slide controller should be a live controller' assert live_controller.is_live is True, 'The slide controller should be a live controller'
class TestPreviewLiveController(TestCase): def test_initial_preview_controller():
def test_initial_preview_controller(self):
""" """
Test the initial preview slide controller state. Test the initial preview slide controller state.
""" """

View File

@ -22,7 +22,7 @@
Package to test the openlp.plugin.bible.lib.https package. Package to test the openlp.plugin.bible.lib.https package.
""" """
import os import os
from unittest import TestCase, skipIf from unittest import TestCase, skipIf, skip
from unittest.mock import MagicMock from unittest.mock import MagicMock
from openlp.core.common.registry import Registry from openlp.core.common.registry import Registry
@ -122,6 +122,7 @@ class TestBibleHTTP(TestCase):
# THEN: We should get back a valid service item # THEN: We should get back a valid service item
assert len(results.verse_list) == 36, 'The book of John should not have had any verses added or removed' assert len(results.verse_list) == 36, 'The book of John should not have had any verses added or removed'
@skip("We can't currently parse BibelServer")
def test_bibleserver_get_bibles(self): def test_bibleserver_get_bibles(self):
""" """
Test getting list of bibles from BibleServer.com Test getting list of bibles from BibleServer.com

View File

@ -35,8 +35,9 @@ class FakeIP4InterfaceEntry(QObject):
""" """
Class to face an interface for testing purposes Class to face an interface for testing purposes
""" """
def __init__(self, name='lo'): def __init__(self, name='lo', is_valid=True):
self.my_name = name self.my_name = name
self._is_valid = is_valid
if name in ['localhost', 'lo']: if name in ['localhost', 'lo']:
self.my_ip = QNetworkAddressEntry() self.my_ip = QNetworkAddressEntry()
self.my_ip.setBroadcast(QHostAddress('255.0.0.0')) self.my_ip.setBroadcast(QHostAddress('255.0.0.0'))
@ -75,7 +76,7 @@ class FakeIP4InterfaceEntry(QObject):
return self.my_name return self.my_name
def isValid(self): def isValid(self):
return True return self._is_valid
class TestInterfaces(TestCase, TestMixin): class TestInterfaces(TestCase, TestMixin):
@ -92,6 +93,7 @@ class TestInterfaces(TestCase, TestMixin):
self.fake_lo = FakeIP4InterfaceEntry() self.fake_lo = FakeIP4InterfaceEntry()
self.fake_localhost = FakeIP4InterfaceEntry(name='localhost') self.fake_localhost = FakeIP4InterfaceEntry(name='localhost')
self.fake_address = FakeIP4InterfaceEntry(name='eth25') self.fake_address = FakeIP4InterfaceEntry(name='eth25')
self.invalid_if = FakeIP4InterfaceEntry(name='invalid', is_valid=False)
def tearDown(self): def tearDown(self):
""" """
@ -126,7 +128,7 @@ class TestInterfaces(TestCase, TestMixin):
# GIVEN: Test environment # GIVEN: Test environment
call_debug = [ call_debug = [
call('Getting local IPv4 interface(es) information'), call('Getting local IPv4 interface(es) information'),
call("Filtering out interfaces we don't care about: lo") call('Filtering out interfaces we don\'t care about: lo')
] ]
# WHEN: get_network_interfaces() is called # WHEN: get_network_interfaces() is called
@ -146,7 +148,7 @@ class TestInterfaces(TestCase, TestMixin):
# GIVEN: Test environment # GIVEN: Test environment
call_debug = [ call_debug = [
call('Getting local IPv4 interface(es) information'), call('Getting local IPv4 interface(es) information'),
call("Filtering out interfaces we don't care about: localhost") call('Filtering out interfaces we don\'t care about: localhost')
] ]
# WHEN: get_network_interfaces() is called # WHEN: get_network_interfaces() is called
@ -190,7 +192,7 @@ class TestInterfaces(TestCase, TestMixin):
# GIVEN: Test environment # GIVEN: Test environment
call_debug = [ call_debug = [
call('Getting local IPv4 interface(es) information'), call('Getting local IPv4 interface(es) information'),
call("Filtering out interfaces we don't care about: lo"), call('Filtering out interfaces we don\'t care about: lo'),
call('Checking for isValid and flags == IsUP | IsRunning'), call('Checking for isValid and flags == IsUP | IsRunning'),
call('Checking address(es) protocol'), call('Checking address(es) protocol'),
call('Checking for protocol == IPv4Protocol'), call('Checking for protocol == IPv4Protocol'),
@ -205,4 +207,26 @@ class TestInterfaces(TestCase, TestMixin):
# THEN: There should be a fake 'eth25' interface # THEN: There should be a fake 'eth25' interface
mock_log.debug.assert_has_calls(call_debug) mock_log.debug.assert_has_calls(call_debug)
assert interfaces == self.fake_address.fake_data, "There should have been only 'eth25' interface listed" assert interfaces == self.fake_address.fake_data, 'There should have been only "eth25" interface listed'
@patch.object(openlp.core.common, 'log')
def test_network_interfaces_invalid(self, mock_log):
"""
Test get_network_interfaces() returns an empty dictionary when there are no valid interfaces
"""
# GIVEN: Test environment
call_debug = [
call('Getting local IPv4 interface(es) information'),
call('Checking for isValid and flags == IsUP | IsRunning')
]
call_warning = [call('No active IPv4 network interfaces detected')]
# WHEN: get_network_interfaces() is called
with patch('openlp.core.common.QNetworkInterface') as mock_network_interface:
mock_network_interface.allInterfaces.return_value = [self.invalid_if]
interfaces = get_network_interfaces()
# THEN: There should be a fake 'eth25' interface
mock_log.debug.assert_has_calls(call_debug)
mock_log.warning.assert_has_calls(call_warning)
assert interfaces == {}, 'There should not be any interfaces listed'