2013-12-29 19:46:04 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2019-04-13 13:00:22 +00:00
|
|
|
##########################################################################
|
|
|
|
# OpenLP - Open Source Lyrics Projection #
|
|
|
|
# ---------------------------------------------------------------------- #
|
2020-01-01 02:53:08 +00:00
|
|
|
# Copyright (c) 2008-2020 OpenLP Developers #
|
2019-04-13 13:00:22 +00:00
|
|
|
# ---------------------------------------------------------------------- #
|
|
|
|
# This program is free software: you can redistribute it and/or modify #
|
|
|
|
# it under the terms of the GNU General Public License as published by #
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or #
|
|
|
|
# (at your option) any later version. #
|
|
|
|
# #
|
|
|
|
# This program is distributed in the hope that it will be useful, #
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of #
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the #
|
|
|
|
# GNU General Public License for more details. #
|
|
|
|
# #
|
|
|
|
# You should have received a copy of the GNU General Public License #
|
|
|
|
# along with this program. If not, see <https://www.gnu.org/licenses/>. #
|
|
|
|
##########################################################################
|
2013-12-29 19:46:04 +00:00
|
|
|
"""
|
|
|
|
This module contains tests for the PdfController
|
|
|
|
"""
|
|
|
|
import os
|
2019-10-18 19:14:22 +00:00
|
|
|
import pytest
|
2019-05-22 06:47:00 +00:00
|
|
|
from pathlib import Path
|
|
|
|
from shutil import rmtree, which
|
2014-03-14 22:08:44 +00:00
|
|
|
from tempfile import mkdtemp
|
2019-10-18 16:54:00 +00:00
|
|
|
from unittest import TestCase, skipIf
|
2017-04-24 05:17:55 +00:00
|
|
|
from unittest.mock import MagicMock, patch
|
|
|
|
|
2015-11-07 00:49:40 +00:00
|
|
|
from PyQt5 import QtCore, QtGui
|
2013-12-29 19:46:04 +00:00
|
|
|
|
2019-03-25 22:16:45 +00:00
|
|
|
from openlp.core.common import is_macosx, is_linux, is_win
|
2017-12-28 08:27:44 +00:00
|
|
|
from openlp.core.common.settings import Settings
|
2017-10-10 07:08:44 +00:00
|
|
|
from openlp.core.display.screens import ScreenList
|
2017-12-28 08:27:44 +00:00
|
|
|
from openlp.plugins.presentations.lib.pdfcontroller import PdfController, PdfDocument
|
2014-03-14 22:08:44 +00:00
|
|
|
from tests.helpers.testmixin import TestMixin
|
2018-10-02 04:39:42 +00:00
|
|
|
from tests.utils.constants import RESOURCE_PATH
|
|
|
|
|
2013-12-29 19:46:04 +00:00
|
|
|
|
|
|
|
__default_settings__ = {
|
2019-05-02 13:12:45 +00:00
|
|
|
'presentations/enable_pdf_program': True,
|
|
|
|
'presentations/pdf_program': None,
|
2015-01-22 15:35:52 +00:00
|
|
|
'presentations/thumbnail_scheme': ''
|
2013-12-29 19:46:04 +00:00
|
|
|
}
|
|
|
|
|
2014-04-03 20:22:10 +00:00
|
|
|
SCREEN = {
|
|
|
|
'primary': False,
|
|
|
|
'number': 1,
|
|
|
|
'size': QtCore.QRect(0, 0, 1024, 768)
|
|
|
|
}
|
2019-10-18 16:54:00 +00:00
|
|
|
IS_CI = 'GITLAB_CI' in os.environ or 'APPVEYOR' in os.environ
|
2013-12-31 12:15:18 +00:00
|
|
|
|
2014-04-04 20:24:11 +00:00
|
|
|
|
2019-03-25 21:26:48 +00:00
|
|
|
def get_screen_resolution():
|
|
|
|
"""
|
|
|
|
Get the screen resolution
|
|
|
|
"""
|
|
|
|
if is_macosx():
|
|
|
|
from AppKit import NSScreen
|
|
|
|
screen_size = NSScreen.mainScreen().frame().size
|
|
|
|
return screen_size.width, screen_size.height
|
2019-03-25 22:16:45 +00:00
|
|
|
elif is_win():
|
|
|
|
from win32api import GetSystemMetrics
|
|
|
|
return GetSystemMetrics(0), GetSystemMetrics(1)
|
|
|
|
elif is_linux():
|
|
|
|
from Xlib.display import Display
|
|
|
|
resolution = Display().screen().root.get_geometry()
|
|
|
|
return resolution.width, resolution.height
|
2019-03-25 21:26:48 +00:00
|
|
|
else:
|
|
|
|
return 1024, 768
|
|
|
|
|
|
|
|
|
2014-03-14 22:08:44 +00:00
|
|
|
class TestPdfController(TestCase, TestMixin):
|
2013-12-29 19:46:04 +00:00
|
|
|
"""
|
|
|
|
Test the PdfController.
|
|
|
|
"""
|
|
|
|
def setUp(self):
|
|
|
|
"""
|
|
|
|
Set up the components need for all tests.
|
|
|
|
"""
|
2014-10-22 20:43:05 +00:00
|
|
|
self.setup_application()
|
2014-03-14 22:08:44 +00:00
|
|
|
self.build_settings()
|
2014-04-03 20:22:10 +00:00
|
|
|
# Mocked out desktop object
|
|
|
|
self.desktop = MagicMock()
|
|
|
|
self.desktop.primaryScreen.return_value = SCREEN['primary']
|
|
|
|
self.desktop.screenCount.return_value = SCREEN['number']
|
|
|
|
self.desktop.screenGeometry.return_value = SCREEN['size']
|
|
|
|
self.screens = ScreenList.create(self.desktop)
|
2013-12-29 19:46:04 +00:00
|
|
|
Settings().extend_default_settings(__default_settings__)
|
2017-11-18 11:23:15 +00:00
|
|
|
self.temp_folder_path = Path(mkdtemp())
|
|
|
|
self.thumbnail_folder_path = Path(mkdtemp())
|
2014-03-13 20:59:10 +00:00
|
|
|
self.mock_plugin = MagicMock()
|
2017-11-18 11:23:15 +00:00
|
|
|
self.mock_plugin.settings_section = self.temp_folder_path
|
2013-12-29 19:46:04 +00:00
|
|
|
|
|
|
|
def tearDown(self):
|
|
|
|
"""
|
|
|
|
Delete all the C++ objects at the end so that we don't have a segfault
|
|
|
|
"""
|
2014-04-03 20:22:10 +00:00
|
|
|
del self.screens
|
|
|
|
self.destroy_settings()
|
2019-05-22 06:47:00 +00:00
|
|
|
rmtree(self.thumbnail_folder_path)
|
|
|
|
rmtree(self.temp_folder_path)
|
2014-04-03 20:22:10 +00:00
|
|
|
|
2016-05-31 21:40:13 +00:00
|
|
|
def test_constructor(self):
|
2013-12-29 19:46:04 +00:00
|
|
|
"""
|
2014-03-04 20:18:14 +00:00
|
|
|
Test the Constructor from the PdfController
|
2013-12-29 19:46:04 +00:00
|
|
|
"""
|
|
|
|
# GIVEN: No presentation controller
|
|
|
|
controller = None
|
|
|
|
|
|
|
|
# WHEN: The presentation controller object is created
|
2014-03-13 20:59:10 +00:00
|
|
|
controller = PdfController(plugin=self.mock_plugin)
|
2013-12-29 19:46:04 +00:00
|
|
|
|
|
|
|
# THEN: The name of the presentation controller should be correct
|
2017-12-24 07:11:30 +00:00
|
|
|
assert 'Pdf' == controller.name, 'The name of the presentation controller should be correct'
|
2013-12-29 19:46:04 +00:00
|
|
|
|
2019-10-18 16:54:00 +00:00
|
|
|
@skipIf(IS_CI, "This is failing on CI, skip until we can figure out what the problem is")
|
2019-05-02 13:12:45 +00:00
|
|
|
def load_pdf(self, exe_path):
|
2013-12-29 19:46:04 +00:00
|
|
|
"""
|
2019-05-02 13:12:45 +00:00
|
|
|
Test loading a Pdf using the PdfController
|
2013-12-29 19:46:04 +00:00
|
|
|
"""
|
|
|
|
# GIVEN: A Pdf-file
|
2017-12-24 08:42:46 +00:00
|
|
|
test_file_path = RESOURCE_PATH / 'presentations' / 'pdf_test1.pdf'
|
2013-12-31 12:15:18 +00:00
|
|
|
|
2013-12-29 19:46:04 +00:00
|
|
|
# WHEN: The Pdf is loaded
|
2019-05-02 13:12:45 +00:00
|
|
|
Settings().setValue('presentations/pdf_program', exe_path)
|
2014-03-13 20:59:10 +00:00
|
|
|
controller = PdfController(plugin=self.mock_plugin)
|
2017-11-18 11:23:15 +00:00
|
|
|
controller.temp_folder = self.temp_folder_path
|
|
|
|
controller.thumbnail_folder = self.thumbnail_folder_path
|
2017-12-24 08:42:46 +00:00
|
|
|
document = PdfDocument(controller, test_file_path)
|
2013-12-29 19:46:04 +00:00
|
|
|
loaded = document.load_presentation()
|
2013-12-31 12:15:18 +00:00
|
|
|
|
2013-12-29 19:46:04 +00:00
|
|
|
# THEN: The load should succeed and we should be able to get a pagecount
|
2017-12-22 21:04:29 +00:00
|
|
|
assert loaded is True, 'The loading of the PDF should succeed.'
|
|
|
|
assert 3 == document.get_slide_count(), 'The pagecount of the PDF should be 3.'
|
2014-04-03 20:22:10 +00:00
|
|
|
|
2019-05-02 13:12:45 +00:00
|
|
|
def load_pdf_pictures(self, exe_path):
|
2014-04-03 20:22:10 +00:00
|
|
|
"""
|
2019-05-02 13:12:45 +00:00
|
|
|
Test loading a Pdf and check the generated pictures' size
|
2014-04-03 20:22:10 +00:00
|
|
|
"""
|
|
|
|
# GIVEN: A Pdf-file
|
2017-12-24 08:42:46 +00:00
|
|
|
test_file_path = RESOURCE_PATH / 'presentations' / 'pdf_test1.pdf'
|
2014-04-03 20:22:10 +00:00
|
|
|
|
|
|
|
# WHEN: The Pdf is loaded
|
2019-05-02 13:12:45 +00:00
|
|
|
Settings().setValue('presentations/pdf_program', exe_path)
|
2014-04-03 20:22:10 +00:00
|
|
|
controller = PdfController(plugin=self.mock_plugin)
|
2017-11-18 11:23:15 +00:00
|
|
|
controller.temp_folder = self.temp_folder_path
|
|
|
|
controller.thumbnail_folder = self.thumbnail_folder_path
|
2017-12-24 08:42:46 +00:00
|
|
|
document = PdfDocument(controller, test_file_path)
|
2014-04-03 20:22:10 +00:00
|
|
|
loaded = document.load_presentation()
|
|
|
|
|
2019-05-02 13:12:45 +00:00
|
|
|
# THEN: The load should succeed and pictures should be created and have been scaled to fit the screen
|
2017-12-22 22:20:04 +00:00
|
|
|
assert loaded is True, 'The loading of the PDF should succeed.'
|
2017-11-18 11:23:15 +00:00
|
|
|
image = QtGui.QImage(os.path.join(str(self.temp_folder_path), 'pdf_test1.pdf', 'mainslide001.png'))
|
2014-04-03 20:22:10 +00:00
|
|
|
# Based on the converter used the resolution will differ a bit
|
|
|
|
if controller.gsbin:
|
2018-12-01 05:52:49 +00:00
|
|
|
assert 1076 == image.height(), 'The height should be 1076'
|
|
|
|
assert 760 == image.width(), 'The width should be 760'
|
2014-04-03 20:22:10 +00:00
|
|
|
else:
|
2019-03-25 21:26:48 +00:00
|
|
|
width, height = get_screen_resolution()
|
2019-03-26 16:45:10 +00:00
|
|
|
# Calculate the width of the PDF based on the aspect ratio of the PDF
|
2019-03-26 17:03:09 +00:00
|
|
|
width = int(round(height * 0.70703125, 0))
|
2019-03-26 16:45:10 +00:00
|
|
|
assert image.height() == height, 'The height should be {height}'.format(height=height)
|
|
|
|
assert image.width() == width, 'The width should be {width}'.format(width=width)
|
2016-05-04 20:58:44 +00:00
|
|
|
|
2019-05-02 13:12:45 +00:00
|
|
|
def test_load_pdf(self):
|
|
|
|
"""
|
|
|
|
Test loading a Pdf with each of the installed backends
|
|
|
|
"""
|
|
|
|
for exe_name in ['gs', 'mutool', 'mudraw']:
|
|
|
|
exe_path = which(exe_name)
|
|
|
|
if exe_path:
|
|
|
|
self.load_pdf(exe_path)
|
|
|
|
self.load_pdf_pictures(exe_path)
|
2019-10-18 19:14:22 +00:00
|
|
|
|
|
|
|
def test_loading_pdf_using_pymupdf(self):
|
|
|
|
try:
|
|
|
|
import fitz # noqa: F401
|
|
|
|
except ImportError:
|
|
|
|
pytest.skip('PyMuPDF is not installed')
|
|
|
|
|
2019-05-02 13:12:45 +00:00
|
|
|
self.load_pdf(None)
|
|
|
|
self.load_pdf_pictures(None)
|
|
|
|
|
2016-05-04 20:58:44 +00:00
|
|
|
@patch('openlp.plugins.presentations.lib.pdfcontroller.check_binary_exists')
|
2016-05-31 21:40:13 +00:00
|
|
|
def test_process_check_binary_mudraw(self, mocked_check_binary_exists):
|
2016-05-04 20:58:44 +00:00
|
|
|
"""
|
|
|
|
Test that the correct output from mudraw is detected
|
|
|
|
"""
|
|
|
|
# GIVEN: A mocked check_binary_exists that returns mudraw output
|
|
|
|
mudraw_output = (b'usage: mudraw [options] input [pages]\n\t-o -\toutput filename (%d for page number)n\t\tsupp'
|
|
|
|
b'orted formats: pgm, ppm, pam, png, pbmn\t-p -\tpasswordn\t-r -\tresolution in dpi (default: '
|
|
|
|
b'72)n\t-w -\twidth (in pixels) (maximum width if -r is specified)n\t-h -\theight (in pixels) '
|
|
|
|
b'(maximum height if -r is specified)')
|
|
|
|
mocked_check_binary_exists.return_value = mudraw_output
|
|
|
|
|
|
|
|
# WHEN: Calling process_check_binary
|
|
|
|
ret = PdfController.process_check_binary('test')
|
|
|
|
|
|
|
|
# THEN: mudraw should be detected
|
2017-12-22 21:04:29 +00:00
|
|
|
assert 'mudraw' == ret, 'mudraw should have been detected'
|
2016-05-04 20:58:44 +00:00
|
|
|
|
|
|
|
@patch('openlp.plugins.presentations.lib.pdfcontroller.check_binary_exists')
|
2016-05-31 21:40:13 +00:00
|
|
|
def test_process_check_binary_new_motool(self, mocked_check_binary_exists):
|
2016-05-04 20:58:44 +00:00
|
|
|
"""
|
|
|
|
Test that the correct output from the new mutool is detected
|
|
|
|
"""
|
|
|
|
# GIVEN: A mocked check_binary_exists that returns new mutool output
|
|
|
|
new_mutool_output = (b'usage: mutool <command> [options]\n\tdraw\t-- convert document\n\trun\t-- run javascript'
|
|
|
|
b'\n\tclean\t-- rewrite pdf file\n\textract\t-- extract font and image resources\n\tinfo\t'
|
|
|
|
b'-- show information about pdf resources\n\tpages\t-- show information about pdf pages\n'
|
|
|
|
b'\tposter\t-- split large page into many tiles\n\tshow\t-- show internal pdf objects\n\t'
|
|
|
|
b'create\t-- create pdf document\n\tmerge\t-- merge pages from multiple pdf sources into a'
|
|
|
|
b'new pdf\n')
|
|
|
|
mocked_check_binary_exists.return_value = new_mutool_output
|
|
|
|
|
|
|
|
# WHEN: Calling process_check_binary
|
|
|
|
ret = PdfController.process_check_binary('test')
|
|
|
|
|
|
|
|
# THEN: mutool should be detected
|
2017-12-22 21:04:29 +00:00
|
|
|
assert 'mutool' == ret, 'mutool should have been detected'
|
2016-05-04 20:58:44 +00:00
|
|
|
|
|
|
|
@patch('openlp.plugins.presentations.lib.pdfcontroller.check_binary_exists')
|
2016-05-31 21:40:13 +00:00
|
|
|
def test_process_check_binary_old_motool(self, mocked_check_binary_exists):
|
2016-05-04 20:58:44 +00:00
|
|
|
"""
|
|
|
|
Test that the output from the old mutool is not accepted
|
|
|
|
"""
|
|
|
|
# GIVEN: A mocked check_binary_exists that returns old mutool output
|
|
|
|
old_mutool_output = (b'usage: mutool <command> [options]\n\tclean\t-- rewrite pdf file\n\textract\t-- extract '
|
|
|
|
b'font and image resources\n\tinfo\t-- show information about pdf resources\n\tposter\t-- '
|
|
|
|
b'split large page into many tiles\n\tshow\t-- show internal pdf objects')
|
|
|
|
mocked_check_binary_exists.return_value = old_mutool_output
|
|
|
|
|
|
|
|
# WHEN: Calling process_check_binary
|
|
|
|
ret = PdfController.process_check_binary('test')
|
|
|
|
|
|
|
|
# THEN: mutool should be detected
|
2017-12-22 21:04:29 +00:00
|
|
|
assert ret is None, 'old mutool should not be accepted!'
|
2016-05-04 20:58:44 +00:00
|
|
|
|
|
|
|
@patch('openlp.plugins.presentations.lib.pdfcontroller.check_binary_exists')
|
2016-05-31 21:40:13 +00:00
|
|
|
def test_process_check_binary_gs(self, mocked_check_binary_exists):
|
2016-05-04 20:58:44 +00:00
|
|
|
"""
|
|
|
|
Test that the correct output from gs is detected
|
|
|
|
"""
|
|
|
|
# GIVEN: A mocked check_binary_exists that returns gs output
|
|
|
|
gs_output = (b'GPL Ghostscript 9.19 (2016-03-23)\nCopyright (C) 2016 Artifex Software, Inc. All rights reserv'
|
|
|
|
b'ed.\nUsage: gs [switches] [file1.ps file2.ps ...]')
|
|
|
|
mocked_check_binary_exists.return_value = gs_output
|
|
|
|
|
|
|
|
# WHEN: Calling process_check_binary
|
|
|
|
ret = PdfController.process_check_binary('test')
|
|
|
|
|
|
|
|
# THEN: mutool should be detected
|
2017-12-22 21:04:29 +00:00
|
|
|
assert 'gs' == ret, 'mutool should have been detected'
|