2014-07-11 12:54:00 +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/>. #
|
|
|
|
##########################################################################
|
2014-07-11 12:54:00 +00:00
|
|
|
"""
|
|
|
|
Functional tests to test the Impress class and related methods.
|
|
|
|
"""
|
2014-07-15 09:23:45 +00:00
|
|
|
import shutil
|
|
|
|
from tempfile import mkdtemp
|
2017-12-28 08:27:44 +00:00
|
|
|
from unittest import TestCase
|
2019-11-03 20:46:21 +00:00
|
|
|
from unittest.mock import MagicMock, call, patch
|
2014-07-15 09:23:45 +00:00
|
|
|
|
2017-09-17 19:43:15 +00:00
|
|
|
from openlp.plugins.presentations.lib.impresscontroller import ImpressController, ImpressDocument, TextType
|
2017-04-24 05:17:55 +00:00
|
|
|
from tests.helpers.testmixin import TestMixin
|
2018-10-02 04:39:42 +00:00
|
|
|
from tests.utils.constants import RESOURCE_PATH
|
2017-04-24 05:17:55 +00:00
|
|
|
|
2014-07-11 12:54:00 +00:00
|
|
|
|
2014-07-15 09:23:45 +00:00
|
|
|
class TestImpressController(TestCase, TestMixin):
|
|
|
|
"""
|
|
|
|
Test the ImpressController Class
|
|
|
|
"""
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
"""
|
|
|
|
Set up the patches and mocks need for all tests.
|
|
|
|
"""
|
2014-10-22 20:43:05 +00:00
|
|
|
self.setup_application()
|
2014-07-15 09:23:45 +00:00
|
|
|
self.build_settings()
|
|
|
|
self.mock_plugin = MagicMock()
|
|
|
|
self.temp_folder = mkdtemp()
|
|
|
|
self.mock_plugin.settings_section = self.temp_folder
|
2014-07-11 13:35:44 +00:00
|
|
|
|
2014-07-15 09:23:45 +00:00
|
|
|
def tearDown(self):
|
|
|
|
"""
|
|
|
|
Stop the patches
|
|
|
|
"""
|
|
|
|
self.destroy_settings()
|
|
|
|
shutil.rmtree(self.temp_folder)
|
2014-07-11 12:54:00 +00:00
|
|
|
|
2016-05-31 21:40:13 +00:00
|
|
|
def test_constructor(self):
|
2014-07-15 09:23:45 +00:00
|
|
|
"""
|
|
|
|
Test the Constructor from the ImpressController
|
|
|
|
"""
|
|
|
|
# GIVEN: No presentation controller
|
|
|
|
controller = None
|
|
|
|
|
|
|
|
# WHEN: The presentation controller object is created
|
|
|
|
controller = ImpressController(plugin=self.mock_plugin)
|
|
|
|
|
|
|
|
# THEN: The name of the presentation controller should be correct
|
2017-12-22 21:04:29 +00:00
|
|
|
assert 'Impress' == controller.name, 'The name of the presentation controller should be correct'
|
2014-07-15 09:23:45 +00:00
|
|
|
|
2017-12-17 15:28:21 +00:00
|
|
|
@patch('openlp.plugins.presentations.lib.impresscontroller.log')
|
|
|
|
def test_check_available(self, mocked_log):
|
|
|
|
"""
|
|
|
|
Test `ImpressController.check_available` on Windows
|
|
|
|
"""
|
|
|
|
# GIVEN: An instance of :class:`ImpressController`
|
|
|
|
controller = ImpressController(plugin=self.mock_plugin)
|
|
|
|
|
|
|
|
# WHEN: `check_available` is called on Windows and `get_com_servicemanager` returns None
|
|
|
|
with patch('openlp.plugins.presentations.lib.impresscontroller.is_win', return_value=True), \
|
|
|
|
patch.object(controller, 'get_com_servicemanager', return_value=None) as mocked_get_com_servicemanager:
|
|
|
|
result = controller.check_available()
|
|
|
|
|
|
|
|
# THEN: `check_available` should return False
|
|
|
|
assert mocked_get_com_servicemanager.called is True
|
|
|
|
assert result is False
|
|
|
|
|
|
|
|
@patch('openlp.plugins.presentations.lib.impresscontroller.log')
|
2019-11-03 20:46:21 +00:00
|
|
|
def test_check_available_on_windows(self, mocked_log):
|
2017-12-17 15:28:21 +00:00
|
|
|
"""
|
|
|
|
Test `ImpressController.check_available` on Windows
|
|
|
|
"""
|
|
|
|
# GIVEN: An instance of :class:`ImpressController`
|
|
|
|
controller = ImpressController(plugin=self.mock_plugin)
|
|
|
|
|
|
|
|
# WHEN: `check_available` is called on Windows and `get_com_servicemanager` returns an object
|
|
|
|
mocked_com_object = MagicMock()
|
|
|
|
with patch('openlp.plugins.presentations.lib.impresscontroller.is_win', return_value=True), \
|
|
|
|
patch.object(controller, 'get_com_servicemanager', return_value=mocked_com_object) \
|
|
|
|
as mocked_get_com_servicemanager:
|
|
|
|
result = controller.check_available()
|
|
|
|
|
|
|
|
# THEN: `check_available` should return True
|
|
|
|
assert mocked_get_com_servicemanager.called is True
|
|
|
|
assert result is True
|
|
|
|
|
|
|
|
@patch('openlp.plugins.presentations.lib.impresscontroller.log')
|
|
|
|
@patch('openlp.plugins.presentations.lib.impresscontroller.is_win', return_value=False)
|
2019-11-03 20:46:21 +00:00
|
|
|
def test_check_available_on_linux(self, mocked_is_win, mocked_log):
|
2017-12-17 15:28:21 +00:00
|
|
|
"""
|
|
|
|
Test `ImpressController.check_available` when not on Windows
|
|
|
|
"""
|
|
|
|
# GIVEN: An instance of :class:`ImpressController`
|
|
|
|
controller = ImpressController(plugin=self.mock_plugin)
|
|
|
|
|
|
|
|
# WHEN: `check_available` is called on Windows and `uno_available` is True
|
|
|
|
with patch('openlp.plugins.presentations.lib.impresscontroller.uno_available', True), \
|
|
|
|
patch.object(controller, 'get_com_servicemanager') as mocked_get_com_servicemanager:
|
|
|
|
result = controller.check_available()
|
|
|
|
|
|
|
|
# THEN: `check_available` should return True
|
|
|
|
assert mocked_get_com_servicemanager.called is False
|
|
|
|
assert result is True
|
|
|
|
|
2019-11-03 20:46:21 +00:00
|
|
|
@patch('openlp.plugins.presentations.lib.impresscontroller.is_win', return_value=True)
|
|
|
|
def test_start_process_on_windows(self, mocked_is_win):
|
|
|
|
"""
|
|
|
|
Test that start_process() on Windows starts the process
|
|
|
|
"""
|
|
|
|
# GIVEN: An ImpressController object
|
|
|
|
controller = ImpressController(plugin=self.mock_plugin)
|
|
|
|
controller.get_com_servicemanager = MagicMock(return_value=MagicMock())
|
|
|
|
|
|
|
|
# WHEN: start_process() is called
|
|
|
|
controller.start_process()
|
|
|
|
|
|
|
|
# THEN: The correct methods should have been called
|
|
|
|
controller.get_com_servicemanager.assert_called_once()
|
|
|
|
assert controller.manager._FlagAsMethod.call_args_list == [call('Bridge_GetStruct'),
|
|
|
|
call('Bridge_GetValueObject')]
|
|
|
|
|
|
|
|
@patch('openlp.plugins.presentations.lib.impresscontroller.is_win', return_value=False)
|
|
|
|
@patch('openlp.plugins.presentations.lib.impresscontroller.get_uno_command', return_value='libreoffice')
|
|
|
|
@patch('openlp.plugins.presentations.lib.impresscontroller.QtCore.QProcess')
|
|
|
|
def test_start_process_on_linux(self, MockQProcess, mocked_get_uno_command, mocked_is_win):
|
|
|
|
"""
|
|
|
|
Test that start_process() on Linux starts the process
|
|
|
|
"""
|
|
|
|
# GIVEN: An ImpressController object
|
|
|
|
mocked_process = MagicMock()
|
|
|
|
MockQProcess.return_value = mocked_process
|
|
|
|
controller = ImpressController(plugin=self.mock_plugin)
|
|
|
|
|
|
|
|
# WHEN: start_process() is called
|
|
|
|
controller.start_process()
|
|
|
|
|
|
|
|
# THEN: The correct methods should have been called
|
|
|
|
mocked_get_uno_command.assert_called_once()
|
|
|
|
MockQProcess.assert_called_once()
|
|
|
|
assert controller.process is mocked_process
|
|
|
|
mocked_process.startDetached.assert_called_once_with('libreoffice')
|
|
|
|
|
2014-07-15 09:23:45 +00:00
|
|
|
|
2014-09-22 21:19:02 +00:00
|
|
|
class TestImpressDocument(TestCase):
|
2014-07-15 09:23:45 +00:00
|
|
|
"""
|
|
|
|
Test the ImpressDocument Class
|
|
|
|
"""
|
2014-07-11 12:54:00 +00:00
|
|
|
def setUp(self):
|
|
|
|
mocked_plugin = MagicMock()
|
|
|
|
mocked_plugin.settings_section = 'presentations'
|
2017-12-24 08:42:46 +00:00
|
|
|
self.file_name = RESOURCE_PATH / 'presentations' / 'test.pptx'
|
2014-07-11 12:54:00 +00:00
|
|
|
self.ppc = ImpressController(mocked_plugin)
|
|
|
|
self.doc = ImpressDocument(self.ppc, self.file_name)
|
|
|
|
|
2016-05-31 21:40:13 +00:00
|
|
|
def test_create_titles_and_notes(self):
|
2014-07-11 12:54:00 +00:00
|
|
|
"""
|
|
|
|
Test ImpressDocument.create_titles_and_notes
|
|
|
|
"""
|
|
|
|
# GIVEN: mocked PresentationController.save_titles_and_notes with
|
|
|
|
# 0 pages and the LibreOffice Document
|
|
|
|
self.doc.save_titles_and_notes = MagicMock()
|
|
|
|
self.doc.document = MagicMock()
|
|
|
|
self.doc.document.getDrawPages.return_value = MagicMock()
|
|
|
|
self.doc.document.getDrawPages().getCount.return_value = 0
|
|
|
|
|
|
|
|
# WHEN reading the titles and notes
|
|
|
|
self.doc.create_titles_and_notes()
|
|
|
|
|
|
|
|
# THEN save_titles_and_notes should have been called with empty arrays
|
|
|
|
self.doc.save_titles_and_notes.assert_called_once_with([], [])
|
|
|
|
|
|
|
|
# GIVEN: reset mock and set it to 2 pages
|
|
|
|
self.doc.save_titles_and_notes.reset_mock()
|
|
|
|
self.doc.document.getDrawPages().getCount.return_value = 2
|
|
|
|
|
|
|
|
# WHEN: a new call to create_titles_and_notes
|
|
|
|
self.doc.create_titles_and_notes()
|
|
|
|
|
|
|
|
# THEN: save_titles_and_notes should have been called once with
|
|
|
|
# two arrays of two elements
|
2018-12-01 05:52:49 +00:00
|
|
|
# self.doc.save_titles_and_notes.assert_called_once_with(['\n', '\n'], [' ', ' '])
|
|
|
|
self.doc.save_titles_and_notes.assert_called_once_with(['', ''], [' ', ' '])
|
2014-07-11 12:54:00 +00:00
|
|
|
|
2016-05-31 21:40:13 +00:00
|
|
|
def test_get_text_from_page_out_of_bound(self):
|
2014-07-11 12:54:00 +00:00
|
|
|
"""
|
|
|
|
Test ImpressDocument.__get_text_from_page with out-of-bounds index
|
|
|
|
"""
|
|
|
|
# GIVEN: mocked LibreOffice Document with one slide,
|
|
|
|
# two notes and three texts
|
|
|
|
self.doc.document = self._mock_a_LibreOffice_document(1, 2, 3)
|
|
|
|
|
|
|
|
# WHEN: __get_text_from_page is called with an index of 0x00
|
|
|
|
result = self.doc._ImpressDocument__get_text_from_page(0, TextType.Notes)
|
|
|
|
|
|
|
|
# THEN: the result should be an empty string
|
2017-12-22 21:04:29 +00:00
|
|
|
assert result == '', 'Result should be an empty string'
|
2014-07-11 12:54:00 +00:00
|
|
|
|
|
|
|
# WHEN: regardless of the type of text, index 0x00 is out of bounds
|
|
|
|
result = self.doc._ImpressDocument__get_text_from_page(0, TextType.Title)
|
|
|
|
|
|
|
|
# THEN: result should be an empty string
|
2017-12-22 21:04:29 +00:00
|
|
|
assert result == '', 'Result should be an empty string'
|
2014-07-11 12:54:00 +00:00
|
|
|
|
|
|
|
# WHEN: when called with 2, it should also be out of bounds
|
|
|
|
result = self.doc._ImpressDocument__get_text_from_page(2, TextType.SlideText)
|
|
|
|
|
|
|
|
# THEN: result should be an empty string ... and, getByIndex should
|
|
|
|
# have never been called
|
2017-12-22 21:04:29 +00:00
|
|
|
assert result == '', 'Result should be an empty string'
|
|
|
|
assert self.doc.document.getDrawPages().getByIndex.call_count == 0, 'There should be no call to getByIndex'
|
2014-07-11 12:54:00 +00:00
|
|
|
|
2016-05-31 21:40:13 +00:00
|
|
|
def test_get_text_from_page_wrong_type(self):
|
2014-07-11 12:54:00 +00:00
|
|
|
"""
|
|
|
|
Test ImpressDocument.__get_text_from_page with wrong TextType
|
|
|
|
"""
|
|
|
|
# GIVEN: mocked LibreOffice Document with one slide, two notes and
|
|
|
|
# three texts
|
|
|
|
self.doc.document = self._mock_a_LibreOffice_document(1, 2, 3)
|
|
|
|
|
|
|
|
# WHEN: called with TextType 3
|
|
|
|
result = self.doc._ImpressDocument__get_text_from_page(1, 3)
|
|
|
|
|
|
|
|
# THEN: result should be an empty string
|
2017-12-22 21:04:29 +00:00
|
|
|
assert result == '', 'Result should be and empty string'
|
|
|
|
assert self.doc.document.getDrawPages().getByIndex.call_count == 0, 'There should be no call to getByIndex'
|
2014-07-11 12:54:00 +00:00
|
|
|
|
2016-05-31 21:40:13 +00:00
|
|
|
def test_get_text_from_page_valid_params(self):
|
2014-07-11 12:54:00 +00:00
|
|
|
"""
|
|
|
|
Test ImpressDocument.__get_text_from_page with valid parameters
|
|
|
|
"""
|
|
|
|
# GIVEN: mocked LibreOffice Document with one slide,
|
|
|
|
# two notes and three texts
|
|
|
|
self.doc.document = self._mock_a_LibreOffice_document(1, 2, 3)
|
|
|
|
|
|
|
|
# WHEN: __get_text_from_page is called to get the Notes
|
|
|
|
result = self.doc._ImpressDocument__get_text_from_page(1, TextType.Notes)
|
|
|
|
|
|
|
|
# THEN: result should be 'Note\nNote\n'
|
2017-12-22 21:04:29 +00:00
|
|
|
assert result == 'Note\nNote\n', 'Result should be \'Note\\n\' times the count of notes in the page'
|
2014-07-11 12:54:00 +00:00
|
|
|
|
|
|
|
# WHEN: get the Title
|
|
|
|
result = self.doc._ImpressDocument__get_text_from_page(1, TextType.Title)
|
|
|
|
|
|
|
|
# THEN: result should be 'Title\n'
|
2017-12-22 21:04:29 +00:00
|
|
|
assert result == 'Title\n', 'Result should be exactly \'Title\\n\''
|
2014-07-11 12:54:00 +00:00
|
|
|
|
|
|
|
# WHEN: get all text
|
|
|
|
result = self.doc._ImpressDocument__get_text_from_page(1, TextType.SlideText)
|
|
|
|
|
|
|
|
# THEN: result should be 'Title\nString\nString\n'
|
2017-12-22 21:04:29 +00:00
|
|
|
assert result == 'Title\nString\nString\n', 'Result should be exactly \'Title\\nString\\nString\\n\''
|
2014-07-11 12:54:00 +00:00
|
|
|
|
|
|
|
def _mock_a_LibreOffice_document(self, page_count, note_count, text_count):
|
2014-07-15 09:23:45 +00:00
|
|
|
"""
|
|
|
|
Helper function, creates a mock libreoffice document.
|
|
|
|
|
|
|
|
:param page_count: Number of pages in the document
|
|
|
|
:param note_count: Number of note pages in the document
|
|
|
|
:param text_count: Number of text pages in the document
|
|
|
|
"""
|
2014-07-11 12:54:00 +00:00
|
|
|
pages = MagicMock()
|
|
|
|
page = MagicMock()
|
|
|
|
pages.getByIndex.return_value = page
|
|
|
|
notes_page = MagicMock()
|
|
|
|
notes_page.getCount.return_value = note_count
|
|
|
|
shape = MagicMock()
|
|
|
|
shape.supportsService.return_value = True
|
|
|
|
shape.getString.return_value = 'Note'
|
|
|
|
notes_page.getByIndex.return_value = shape
|
|
|
|
page.getNotesPage.return_value = notes_page
|
|
|
|
page.getCount.return_value = text_count
|
|
|
|
page.getByIndex.side_effect = self._get_page_shape_side_effect
|
|
|
|
pages.getCount.return_value = page_count
|
|
|
|
document = MagicMock()
|
|
|
|
document.getDrawPages.return_value = pages
|
|
|
|
document.getByIndex.return_value = page
|
|
|
|
return document
|
|
|
|
|
|
|
|
def _get_page_shape_side_effect(*args):
|
2014-07-15 09:23:45 +00:00
|
|
|
"""
|
|
|
|
Helper function.
|
|
|
|
"""
|
2014-07-11 12:54:00 +00:00
|
|
|
page_shape = MagicMock()
|
|
|
|
page_shape.supportsService.return_value = True
|
|
|
|
if args[1] == 0:
|
|
|
|
page_shape.getShapeType.return_value = 'com.sun.star.presentation.TitleTextShape'
|
|
|
|
page_shape.getString.return_value = 'Title'
|
|
|
|
else:
|
|
|
|
page_shape.getString.return_value = 'String'
|
|
|
|
return page_shape
|