# -*- coding: utf-8 -*-

##########################################################################
# OpenLP - Open Source Lyrics Projection                                 #
# ---------------------------------------------------------------------- #
# Copyright (c) 2008-2021 OpenLP Developers                              #
# ---------------------------------------------------------------------- #
# 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/>. #
##########################################################################

"""
This module contains tests for the CCLI SongSelect importer.
It needs re-writing at some point to load real HTML pages from disk and
then test the behaviour based on those. That way if and when CCLI change
their page layout, changing the tests would just be a case of
re-downloading the HTML pages and changing the code to use the new layout.
"""
from unittest import TestCase
from unittest.mock import MagicMock, patch, sentinel

from PyQt5 import QtWidgets, QtCore

from openlp.core.common.registry import Registry
from openlp.plugins.songs.forms.songselectform import SongSelectForm
from openlp.plugins.songs.lib import Song
from openlp.plugins.songs.lib.songselect import BASE_URL, LOGIN_PAGE, Pages, SongSelectImport
from tests.helpers.songfileimport import SongImportTestHelper
from tests.helpers.testmixin import TestMixin
from tests.utils.constants import RESOURCE_PATH


TEST_PATH = RESOURCE_PATH / 'songs' / 'songselect'


class TestSongSelectImport(TestCase, TestMixin):
    """
    Test the :class:`~openlp.plugins.songs.lib.songselect.SongSelectImport` class
    """
    def test_constructor(self):
        """
        Test that constructing a basic SongSelectImport object works correctly
        """
        # GIVEN: The SongSelectImporter class and a mocked out build_opener
        # WHEN: An object is instantiated
        importer = SongSelectImport(sentinel.db_manager, sentinel.webview)

        # THEN: The object should have the correct properties
        assert importer.db_manager is sentinel.db_manager, 'The db_manager should be set'
        assert importer.webview is sentinel.webview, 'The webview should be set'

    def test_get_page_type_login(self):
        """
        Test get_page_type to spot the login page
        """
        # GIVEN: A importer object, and a mocked url
        importer = SongSelectImport(None, None)
        url = QtCore.QUrl('https://profile.ccli.com/account/signin?appContext=SongSelect&'
                          'returnUrl=https%3a%2f%2fsongselect.ccli.com%2f')
        page = MagicMock(url=MagicMock(return_value=url))
        importer.webview = MagicMock(page=MagicMock(return_value=page))

        # WHEN: The method is run
        result = importer.get_page_type()

        # THEN: The correct type should be returned
        assert result == Pages.Login

    def test_get_page_type_home(self):
        """
        Test get_page_type to spot the home page
        """
        # GIVEN: A importer object, and a mocked url
        importer = SongSelectImport(None, None)
        url = QtCore.QUrl('https://songselect.ccli.com')
        page = MagicMock(url=MagicMock(return_value=url))
        importer.webview = MagicMock(page=MagicMock(return_value=page))

        # WHEN: The method is run
        result = importer.get_page_type()

        # THEN: The correct type should be returned
        assert result == Pages.Home

    def test_get_page_type_search(self):
        """
        Test get_page_type to spot the search page
        """
        # GIVEN: A importer object, and a mocked url
        importer = SongSelectImport(None, None)
        url = QtCore.QUrl('https://songselect.ccli.com/search/results?SearchText=test')
        page = MagicMock(url=MagicMock(return_value=url))
        importer.webview = MagicMock(page=MagicMock(return_value=page))

        # WHEN: The method is run
        result = importer.get_page_type()

        # THEN: The correct type should be returned
        assert result == Pages.Search

    def test_get_page_type_song(self):
        """
        Test get_page_type to spot the login page
        """
        # GIVEN: A importer object, and a mocked url
        importer = SongSelectImport(None, None)
        url = QtCore.QUrl('https://songselect.ccli.com/Songs/7115744/song_name/view_lyrics')
        page = MagicMock(url=MagicMock(return_value=url))
        importer.webview = MagicMock(page=MagicMock(return_value=page))

        # WHEN: The method is run
        result = importer.get_page_type()

        # THEN: The correct type should be returned
        assert result == Pages.Song

    def test_get_page_type_other(self):
        """
        Test get_page_type to spot the login page
        """
        # GIVEN: A importer object, and a mocked url
        importer = SongSelectImport(None, None)
        url = QtCore.QUrl('https://openlp.org')
        page = MagicMock(url=MagicMock(return_value=url))
        importer.webview = MagicMock(page=MagicMock(return_value=page))

        # WHEN: The method is run
        result = importer.get_page_type()

        # THEN: The correct type should be returned
        assert result == Pages.Other

    @patch('openlp.plugins.songs.lib.songselect.wait_for')
    def test_run_javascript(self, mocked_wait_for):
        """
        Test run javascript calls the page object
        """
        # GIVEN: A importer object and mocked run js fn
        def runJs(script, handle_result):
            handle_result('processed_{}'.format(script))
        importer = SongSelectImport(None, None)
        importer.webview = MagicMock()
        page = MagicMock()
        page.runJavaScript = runJs
        importer.webview.page.return_value = page

        # WHEN: The set login field method is called
        result = importer._run_javascript('2 + 2')

        # THEN: The javascript should have been called on the page object
        assert result == 'processed_2 + 2'

    def test_reset_webview(self):
        """
        Check that the setUrl method is called when the reset webview method is called
        """
        # GIVEN: A importer object and mock webview
        importer = SongSelectImport(None, None)
        importer.webview = MagicMock()

        # WHEN: The reset_webview method is called
        importer.reset_webview()

        # THEN: The setUrl function should have been called
        importer.webview.setUrl.assert_called_with(QtCore.QUrl(LOGIN_PAGE))

    @patch('openlp.plugins.songs.lib.songselect.SongSelectImport.set_page')
    def test_set_home_page(self, mocked_set_page):
        """
        Test that when the home method is called, it attempts to go to the home page
        """
        # GIVEN: A importer object
        importer = SongSelectImport(None, None)

        # WHEN: The home method is called
        importer.set_home_page()

        # THEN: set_page is called once with the base url
        mocked_set_page.assert_called_with(BASE_URL)

    @patch('openlp.plugins.songs.lib.songselect.SongSelectImport._run_javascript')
    def test_set_page(self, mocked_run_js):
        """
        Test set page runs the correct script
        """
        # GIVEN: A importer object
        importer = SongSelectImport(None, None)

        # WHEN: The set login field method is called
        importer.set_page('my_new_page')

        # THEN: The javascript called should contain the correct values
        mocked_run_js.assert_called_with('document.location = "my_new_page"')

    @patch('openlp.plugins.songs.lib.songselect.SongSelectImport._run_javascript')
    def test_set_login_fields(self, mocked_run_js):
        """
        Test correct js is sent to set login fields
        """
        # GIVEN: A importer object
        importer = SongSelectImport(None, None)

        # WHEN: The set login field method is called
        importer.set_login_fields('my_username', 'my_password')

        # THEN: The javascript called should contain the correct values
        mocked_run_js.assert_called_with(('document.getElementById("EmailAddress").value = "my_username";'
                                          'document.getElementById("Password").value = "my_password";'
                                          ))

    @patch('openlp.plugins.songs.lib.songselect.SongSelectImport._run_javascript')
    @patch('openlp.plugins.songs.lib.songselect.wait_for')
    def test_get_page(self, mocked_wait_for, mocked_run_js):
        """
        Test get page sends js requests
        """
        # GIVEN: A importer object
        importer = SongSelectImport(None, None)
        mocked_run_js.return_value = True

        # WHEN: The get page method is called
        importer.get_page("https://example.com")

        # THEN: The javascript should be run
        assert mocked_run_js.call_count == 2, 'Should be called once for request and once for fetch'
        mocked_wait_for.assert_called_once()

    def test_get_song_number_from_url(self):
        """
        Test the ccli number can be correctly obtained from a url
        """
        # GIVEN: A importer object
        importer = SongSelectImport(None, None)

        # WHEN: The function is called with a valid url
        result = importer.get_song_number_from_url('https://songselect.ccli.com/Songs/7115744/way-maker')

        # THEN: The ccli number is returned
        assert result == '7115744', 'Should have found the ccli number from the url'

    def test_get_song_number_from_url_nonumber(self):
        """
        Test the ccli number function returns None when no number is found
        """
        # GIVEN: A importer object
        importer = SongSelectImport(None, None)

        # WHEN: The function is called with a valid url
        result = importer.get_song_number_from_url('https://songselect.ccli.com/search/results?SearchText=song+7115744')

        # THEN: The returned value should be None
        assert result is None

    @patch('openlp.plugins.songs.lib.songselect.SongSelectImport.get_song_number_from_url')
    @patch('openlp.plugins.songs.lib.songselect.SongSelectImport.get_page')
    def test_get_song_page_raises_exception(self, mocked_get_page, mock_get_num):
        """
        Test that when BeautifulSoup gets a bad song page the get_song() method returns None
        """
        # GIVEN: A mocked callback and an importer object
        mocked_get_page.side_effect = None
        mocked_callback = MagicMock()
        importer = SongSelectImport(None, MagicMock())

        # WHEN: get_song is called
        result = importer.get_song(callback=mocked_callback)

        # THEN: The callback should have been called once and None should be returned
        mocked_callback.assert_called_with()
        assert result is None, 'The get_song() method should have returned None'

    @patch('openlp.plugins.songs.lib.songselect.SongSelectImport.get_song_number_from_url')
    @patch('openlp.plugins.songs.lib.songselect.SongSelectImport.get_page')
    @patch('openlp.plugins.songs.lib.songselect.BeautifulSoup')
    def test_get_song_lyrics_raise_exception(self, MockedBeautifulSoup, mocked_get_page, mock_get_num):
        """
        Test that when BeautifulSoup gets a bad lyrics page the get_song() method returns None
        """
        # GIVEN: A bunch of mocked out stuff and an importer object
        song_page = MagicMock(return_value={'href': '/lyricpage'})
        MockedBeautifulSoup.side_effect = [song_page, TypeError('Test Error')]
        mocked_callback = MagicMock()
        importer = SongSelectImport(None, MagicMock())

        # WHEN: get_song is called
        result = importer.get_song(callback=mocked_callback)

        # THEN: The callback should have been called twice and None should be returned
        assert 2 == mocked_callback.call_count, 'The callback should have been called twice'
        assert result is None, 'The get_song() method should have returned None'

    @patch('openlp.plugins.songs.lib.songselect.log.exception')
    @patch('openlp.plugins.songs.lib.songselect.SongSelectImport.get_song_number_from_url')
    @patch('openlp.plugins.songs.lib.songselect.SongSelectImport.get_page')
    def test_get_song_no_access(self, mocked_get_page, mock_get_num, mock_log_exception):
        """
        Test that the get_song() handles the case when the user's CCLI account has no access to the song
        """
        fake_song_page = '''<!DOCTYPE html><html><body>
        <div class="content-title">
          <h1>Song Title</h1>
          <ul class="authors">
            <li><a>Author 1</a></li>
            <li><a>Author 2</a></li>
          </ul>
        </div>
        <div class="song-content-data"><ul><li><strong>1234_cclinumber_5678</strong></li></ul></div>
        <section class="page-section">
          <a title="View song lyrics" data-open="ssUpgradeModal"></a>
        </section>
        <ul class="song-meta-list">
          <li>Themes</li><li><a>theme1</a></li><li><a>theme2</a></li>
        </ul>
        </body></html>
        '''
        fake_lyrics_page = '''<!DOCTYPE html><html><body>
        <div class="song-viewer lyrics">
            <h3>Verse 1</h3>
            <p>verse thing 1<br>line 2</p>
            <h3>Verse 2</h3>
            <p>verse thing 2</p>
        </div>
        <ul class="copyright">
          <li>Copy thing</li><li>Copy thing 2</li>
        </ul>
        </body></html>
        '''
        mocked_get_page.side_effect = [fake_song_page, fake_lyrics_page]
        mocked_callback = MagicMock()
        importer = SongSelectImport(None, MagicMock())

        # WHEN: get_song is called
        result = importer.get_song(callback=mocked_callback)

        # THEN: None should be returned
        assert result is None, 'The get_song() method should have returned None'

    @patch('openlp.plugins.songs.lib.songselect.SongSelectImport.get_song_number_from_url')
    @patch('openlp.plugins.songs.lib.songselect.SongSelectImport.get_page')
    def test_get_song(self, mocked_get_page, mock_get_num):
        """
        Test that the get_song() method returns the correct song details
        """
        fake_song_page = '''<!DOCTYPE html><html><body>
        <div class="content-title">
          <h1>Song Title</h1>
          <ul class="authors">
            <li><a>Author 1</a></li>
            <li><a>Author 2</a></li>
          </ul>
        </div>
        <div class="song-content-data"><ul><li><strong>1234_cclinumber_5678</strong></li></ul></div>
        <section class="page-section">
          <a title="View song lyrics" href="pretend link"></a>
        </section>
        <ul class="song-meta-list">
          <li>Themes</li><li><a>theme1</a></li><li><a>theme2</a></li>
        </ul>
        </body></html>
        '''
        fake_lyrics_page = '''<!DOCTYPE html><html><body>
        <div class="song-viewer lyrics">
            <h3>Verse 1</h3>
            <p>verse thing 1<br>line 2</p>
            <h3>Verse 2</h3>
            <p>verse thing 2</p>
            <h3>Spoken Words</h3>
            <p>completely custom verse type</p>
        </div>
        <ul class="copyright">
          <li>Copy thing</li><li>Copy thing 2</li>
        </ul>
        </body></html>
        '''
        mocked_get_page.side_effect = [fake_song_page, fake_lyrics_page]
        mocked_callback = MagicMock()
        importer = SongSelectImport(None, MagicMock())

        # WHEN: get_song is called
        result = importer.get_song(callback=mocked_callback)

        # THEN: The callback should have been called three times and the song should be returned
        assert 3 == mocked_callback.call_count, 'The callback should have been called twice'
        assert result is not None, 'The get_song() method should have returned a song dictionary'
        assert result['title'] == 'Song Title'
        assert result['authors'] == ['Author 1', 'Author 2']
        assert result['copyright'] == 'Copy thing/Copy thing 2'
        assert result['topics'] == ['theme1', 'theme2']
        assert result['ccli_number'] == '1234_cclinumber_5678'
        assert result['verses'] == [{'label': 'Verse 1', 'lyrics': 'verse thing 1\nline 2'},
                                    {'label': 'Verse 2', 'lyrics': 'verse thing 2'},
                                    {'label': 'Spoken Words', 'lyrics': 'completely custom verse type'}]

    @patch('openlp.plugins.songs.lib.songselect.clean_song')
    @patch('openlp.plugins.songs.lib.songselect.Topic')
    @patch('openlp.plugins.songs.lib.songselect.Author')
    def test_save_song_new_author(self, MockedAuthor, MockedTopic, mocked_clean_song):
        """
        Test that saving a song with a new author performs the correct actions
        """
        # GIVEN: A song to save, and some mocked out objects
        song_dict = {
            'title': 'Arky Arky',
            'authors': ['Public Domain'],
            'verses': [
                {'label': 'Verse 1', 'lyrics': 'The Lord told Noah: there\'s gonna be a floody, floody'},
                {'label': 'Chorus 1', 'lyrics': 'So, rise and shine, and give God the glory, glory'},
                {'label': 'Verse 2', 'lyrics': 'The Lord told Noah to build him an arky, arky'}
            ],
            'copyright': 'Public Domain',
            'ccli_number': '123456'
        }
        MockedAuthor.display_name.__eq__.return_value = False
        MockedTopic.name.__eq__.return_value = False
        mocked_db_manager = MagicMock()
        mocked_db_manager.get_object_filtered.return_value = None
        importer = SongSelectImport(mocked_db_manager, MagicMock())

        # WHEN: The song is saved to the database
        result = importer.save_song(song_dict)

        # THEN: The return value should be a Song class and the mocked_db_manager should have been called
        assert isinstance(result, Song), 'The returned value should be a Song object'
        mocked_clean_song.assert_called_with(mocked_db_manager, result)
        assert 2 == mocked_db_manager.save_object.call_count, \
            'The save_object() method should have been called twice'
        mocked_db_manager.get_object_filtered.assert_called_with(MockedAuthor, False)
        MockedAuthor.populate.assert_called_with(first_name='Public', last_name='Domain', display_name='Public Domain')
        assert 1 == len(result.authors_songs), 'There should only be one author'

    @patch('openlp.plugins.songs.lib.songselect.clean_song')
    @patch('openlp.plugins.songs.lib.songselect.Author')
    def test_save_song_existing_author(self, MockedAuthor, mocked_clean_song):
        """
        Test that saving a song with an existing author performs the correct actions
        """
        # GIVEN: A song to save, and some mocked out objects
        song_dict = {
            'title': 'Arky Arky',
            'authors': ['Public Domain'],
            'verses': [
                {'label': 'Verse 1', 'lyrics': 'The Lord told Noah: there\'s gonna be a floody, floody'},
                {'label': 'Chorus 1', 'lyrics': 'So, rise and shine, and give God the glory, glory'},
                {'label': 'Verse 2', 'lyrics': 'The Lord told Noah to build him an arky, arky'}
            ],
            'copyright': 'Public Domain',
            'ccli_number': '123456'
        }
        MockedAuthor.display_name.__eq__.return_value = False
        mocked_db_manager = MagicMock()
        mocked_db_manager.get_object_filtered.return_value = MagicMock()
        importer = SongSelectImport(mocked_db_manager, MagicMock())

        # WHEN: The song is saved to the database
        result = importer.save_song(song_dict)

        # THEN: The return value should be a Song class and the mocked_db_manager should have been called
        assert isinstance(result, Song), 'The returned value should be a Song object'
        mocked_clean_song.assert_called_with(mocked_db_manager, result)
        assert 2 == mocked_db_manager.save_object.call_count, \
            'The save_object() method should have been called twice'
        mocked_db_manager.get_object_filtered.assert_called_with(MockedAuthor, False)
        assert 0 == MockedAuthor.populate.call_count, 'A new author should not have been instantiated'
        assert 1 == len(result.authors_songs), 'There should only be one author'

    @patch('openlp.plugins.songs.lib.songselect.clean_song')
    @patch('openlp.plugins.songs.lib.songselect.Author')
    def test_save_song_unknown_author(self, MockedAuthor, mocked_clean_song):
        """
        Test that saving a song with an author name of only one word performs the correct actions
        """
        # GIVEN: A song to save, and some mocked out objects
        song_dict = {
            'title': 'Arky Arky',
            'authors': ['Unknown'],
            'verses': [
                {'label': 'Verse 1', 'lyrics': 'The Lord told Noah: there\'s gonna be a floody, floody'},
                {'label': 'Chorus 1', 'lyrics': 'So, rise and shine, and give God the glory, glory'},
                {'label': 'Verse 2', 'lyrics': 'The Lord told Noah to build him an arky, arky'}
            ],
            'copyright': 'Public Domain',
            'ccli_number': '123456'
        }
        MockedAuthor.display_name.__eq__.return_value = False
        mocked_db_manager = MagicMock()
        mocked_db_manager.get_object_filtered.return_value = None
        importer = SongSelectImport(mocked_db_manager, MagicMock())

        # WHEN: The song is saved to the database
        result = importer.save_song(song_dict)

        # THEN: The return value should be a Song class and the mocked_db_manager should have been called
        assert isinstance(result, Song), 'The returned value should be a Song object'
        mocked_clean_song.assert_called_with(mocked_db_manager, result)
        assert 2 == mocked_db_manager.save_object.call_count, \
            'The save_object() method should have been called twice'
        mocked_db_manager.get_object_filtered.assert_called_with(MockedAuthor, False)
        MockedAuthor.populate.assert_called_with(first_name='Unknown', last_name='', display_name='Unknown')
        assert 1 == len(result.authors_songs), 'There should only be one author'

    @patch('openlp.plugins.songs.lib.songselect.clean_song')
    @patch('openlp.plugins.songs.lib.songselect.Topic')
    @patch('openlp.plugins.songs.lib.songselect.Author')
    def test_save_song_topics(self, MockedAuthor, MockedTopic, mocked_clean_song):
        """
        Test that saving a song with topics performs the correct actions
        Also check that a verse with no number is retitled to 1
        """
        # GIVEN: A song to save, and some mocked out objects
        song_dict = {
            'title': 'Arky Arky',
            'authors': ['Public Domain'],
            'verses': [
                {'label': 'Verse', 'lyrics': 'The Lord told Noah: there\'s gonna be a floody, floody'},
                {'label': 'Chorus 1', 'lyrics': 'So, rise and shine, and give God the glory, glory'},
                {'label': 'Verse 2', 'lyrics': 'The Lord told Noah to build him an arky, arky'}
            ],
            'copyright': 'Public Domain',
            'ccli_number': '123456',
            'topics': ['Old Testement', 'Flood']
        }

        def save_object(b):
            b.topics = []
        MockedAuthor.display_name.__eq__.return_value = False
        MockedTopic.name.__eq__.return_value = False
        mocked_db_manager = MagicMock()
        mocked_db_manager.get_object_filtered.return_value = None
        mocked_db_manager.save_object = save_object
        importer = SongSelectImport(mocked_db_manager, MagicMock())

        # WHEN: The song is saved to the database
        result = importer.save_song(song_dict)

        # THEN: The return value should be a Song class and the topics should have been added
        assert isinstance(result, Song), 'The returned value should be a Song object'
        mocked_clean_song.assert_called_with(mocked_db_manager, result)
        assert MockedTopic.populate.call_count == 2, 'Should have created 2 new topics'
        MockedTopic.populate.assert_called_with(name='Flood')
        assert 1 == len(result.authors_songs), 'There should only be one author'


class TestSongSelectForm(TestCase, TestMixin):
    """
    Test the :class:`~openlp.plugins.songs.forms.songselectform.SongSelectForm` class
    """
    def setUp(self):
        """
        Some set up for this test suite
        """
        self.setup_application()
        self.app.setApplicationVersion('0.0')
        self.app.process_events = lambda: None
        Registry.create()
        Registry().register('application', self.app)
        Registry().register('settings', MagicMock())
        self.grid_patcher = patch('openlp.plugins.songs.forms.songselectdialog.QtWidgets.QGridLayout')
        self.web_patcher = patch('openlp.plugins.songs.forms.songselectdialog.WebEngineView')
        self.vbox_patcher = patch('openlp.plugins.songs.forms.songselectdialog.QtWidgets.QVBoxLayout')
        self.grid_patcher.start()
        self.web_patcher.start()
        self.vbox_patcher.start()

    def tearDown(self):
        self.grid_patcher.stop()
        self.web_patcher.stop()
        self.vbox_patcher.stop()

    def test_create_form(self):
        """
        Test that we can create the SongSelect form
        """
        # GIVEN: The SongSelectForm class and a mocked db manager
        mocked_plugin = MagicMock()
        mocked_db_manager = MagicMock()

        # WHEN: We create an instance
        ssform = SongSelectForm(None, mocked_plugin, mocked_db_manager)

        # THEN: The correct properties should have been assigned
        assert mocked_plugin == ssform.plugin, 'The correct plugin should have been assigned'
        assert mocked_db_manager == ssform.db_manager, 'The correct db_manager should have been assigned'

    @patch('openlp.plugins.songs.lib.songselect.SongSelectImport')
    def test_initialise(self, mocked_ss_import):
        """
        Test the initialise method
        """
        # GIVEN: The SongSelectForm
        ssform = SongSelectForm(None, MagicMock(), MagicMock())

        # WHEN: The initialise method is run
        ssform.initialise()

        # THEN: The import object should exist, song var should be None, and the page hooked up
        assert ssform.song is None
        assert isinstance(ssform.song_select_importer, SongSelectImport), 'SongSelectImport object should be created'
        assert ssform.webview.page.call_count == 2, 'Page should be called twice, once for each load handler'

    @patch('openlp.plugins.songs.forms.songselectform.QtWidgets.QDialog.exec')
    def test_exec(self, mocked_exec):
        """
        Test the exec method
        """
        # GIVEN: The SongSelectForm
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        ssform.song_select_importer = MagicMock()
        ssform.stacked_widget = MagicMock()

        # WHEN: The initialise method is run
        ssform.exec()

        # THEN: Should have reset webview, set stack to 0 and pass on the event
        ssform.song_select_importer.reset_webview.assert_called_once()
        ssform.stacked_widget.setCurrentIndex.assert_called_with(0)
        mocked_exec.assert_called_once()

    @patch('openlp.plugins.songs.forms.songselectform.QtWidgets.QDialog.done')
    @patch('openlp.plugins.songs.forms.songselectform.QtWidgets.QProgressDialog')
    def test_done(self, mocked_prog_dialog, mocked_done):
        """
        Test the done method closes th dialog
        """
        # GIVEN: The SongSelectForm
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        ssform.song_select_importer = MagicMock()

        # WHEN: The initialise method is run
        ssform.done('result')

        # THEN: Should have passed on the event
        mocked_done.assert_called_once()

    def test_page_load_started(self):
        """
        Test the page_load_started method
        """
        # GIVEN: The SongSelectForm
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        ssform.song_progress_bar = MagicMock()
        ssform.import_button = MagicMock()
        ssform.view_button = MagicMock()
        ssform.back_button = MagicMock()
        ssform.url_bar = MagicMock()
        ssform.message_area = MagicMock()

        # WHEN: The method is run
        ssform.page_load_started()

        # THEN: The UI should be set up accordingly (working bar and disabled buttons)
        ssform.song_progress_bar.setMaximum.assert_called_with(0)
        ssform.song_progress_bar.setVisible.assert_called_with(True)
        ssform.import_button.setEnabled.assert_called_with(False)
        ssform.view_button.setEnabled.assert_called_with(False)
        ssform.back_button.setEnabled.assert_called_with(False)
        ssform.message_area.setText.assert_called_with('')

    def test_page_loaded_login(self):
        """
        Test the page_loaded method for a "Login" page
        """
        # GIVEN: The SongSelectForm and mocked login page
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        ssform.song_select_importer = MagicMock()
        ssform.song_select_importer.get_page_type.return_value = Pages.Login
        ssform.signin_page_loaded = MagicMock()
        ssform.url_bar = MagicMock()

        # WHEN: The method is run
        ssform.page_loaded(True)

        # THEN: The signin page method should be called
        ssform.signin_page_loaded.assert_called_once()

    def test_page_loaded_song(self):
        """
        Test the page_loaded method for a "Song" page
        """
        # GIVEN: The SongSelectForm and mocked song page
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        ssform.song_select_importer = MagicMock()
        ssform.song_select_importer.get_page_type.return_value = Pages.Song
        ssform.song_progress_bar = MagicMock()
        ssform.url_bar = MagicMock()

        # WHEN: The method is run
        ssform.page_loaded(True)

        # THEN: Progress bar should have been set max 3 (for loading song)
        ssform.song_progress_bar.setMaximum.assert_called_with(3)
        ssform.song_progress_bar.setVisible.call_count == 2

    @patch('openlp.plugins.songs.forms.songselectform.translate')
    def test_page_loaded_song_no_access(self, mocked_translate):
        """
        Test the page_loaded method for a "Song" page to which the CCLI account has no access
        """
        # GIVEN: The SongSelectForm and mocked song page and translate function
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        ssform.song_select_importer = MagicMock()
        ssform.song_select_importer.get_page_type.return_value = Pages.Song
        ssform.song_select_importer.get_song.return_value = None
        ssform.song_progress_bar = MagicMock()
        ssform.url_bar = MagicMock()
        ssform.message_area = MagicMock()
        mocked_translate.return_value = 'some message'

        # WHEN: The method is run
        ssform.page_loaded(True)

        # THEN: The no access message should be shown and the progress bar should be less than 3
        ssform.message_area.setText.assert_called_with('some message')
        ssform.song_progress_bar.setValue.call_count < 4

    def test_page_loaded_other(self):
        """
        Test the page_loaded method for an "Other" page
        """
        # GIVEN: The SongSelectForm and mocked other page
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        ssform.song_select_importer = MagicMock()
        ssform.song_select_importer.get_page_type.return_value = Pages.Other
        ssform.song_progress_bar = MagicMock()
        ssform.back_button = MagicMock()
        ssform.url_bar = MagicMock()

        # WHEN: The method is run
        ssform.page_loaded(True)

        # THEN: Back button should be available
        ssform.back_button.setEnabled.assert_called_with(True)

    def test_signin_page_loaded(self):
        """
        Test that the signin_page_loaded method calls the appropriate method
        """
        # GIVEN: The SongSelectForm and mocked settings
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        ssform.song_select_importer = MagicMock()
        ssform.settings.value = MagicMock(side_effect=['user', 'pass'])

        # WHEN: The method is run
        ssform.signin_page_loaded()

        # THEN: Correct values should have been sent from the settings
        ssform.song_select_importer.set_login_fields.assert_called_with('user', 'pass')

    @patch('openlp.plugins.songs.forms.songselectdialog.QtWidgets.QListWidgetItem')
    def test_view_song(self, mock_qtlist):
        """
        Test that the _view_song method does the important stuff
        """
        # GIVEN: The SongSelectForm, mocks and a song
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        ssform.stacked_widget = MagicMock()
        ssform.title_edit = MagicMock()
        ssform.copyright_edit = MagicMock()
        ssform.ccli_edit = MagicMock()
        ssform.lyrics_table_widget = MagicMock()
        ssform.author_list_widget = MagicMock()
        ssform.song = {
            'title': 'Song Title',
            'copyright': 'copy thing',
            'ccli_number': '1234',
            'authors': ['Bob', 'Jo'],
            'verses': [{'lyrics': 'hello', 'label': 'Verse 1'}]
        }

        # WHEN: The method is run
        ssform._view_song()

        # THEN: Page should have changed in the stacked widget and ui should have been updated
        ssform.stacked_widget.setCurrentIndex.assert_called_with(1)
        ssform.title_edit.setText.assert_called_with('Song Title')
        ssform.copyright_edit.setText.assert_called_with('copy thing')
        ssform.ccli_edit.setText.assert_called_with('1234')
        assert ssform.lyrics_table_widget.setItem.call_count > 0
        assert ssform.author_list_widget.addItem.call_count > 0

    @patch('openlp.plugins.songs.forms.songselectform.QtWidgets.QMessageBox.critical')
    def test_view_song_invalid(self, mock_message):
        """
        Test that the _view_song doesn't mess up when the song doesn't exist
        """
        # GIVEN: The SongSelectForm, mocks and a song
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        ssform.stacked_widget = MagicMock()
        ssform.song = None

        # WHEN: The method is run
        ssform._view_song()

        # THEN: Page should not have changed and a warning should show
        assert ssform.stacked_widget.setCurrentIndex.call_count == 0
        mock_message.assert_called_once()

    def test_on_url_bar_return_pressed(self):
        """
        Test that the on_url_bar_return_pressed method changes the page
        """
        # GIVEN: The SongSelectForm, mocks and a song
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        ssform.url_bar = MagicMock()
        ssform.url_bar.text.return_value = "test"
        ssform.song_select_importer = MagicMock()

        # WHEN: The method is run
        ssform.on_url_bar_return_pressed()

        # THEN: Page should not have changed and a warning should show
        ssform.song_select_importer.set_page.assert_called_with("test")

    @patch('openlp.plugins.songs.forms.songselectform.and_')
    @patch('openlp.plugins.songs.forms.songselectform.Song')
    @patch('openlp.plugins.songs.forms.songselectform.QtWidgets.QMessageBox.information')
    @patch('openlp.plugins.songs.forms.songselectform.QtWidgets.QMessageBox.question')
    @patch('openlp.plugins.songs.forms.songselectform.translate')
    def test_on_import(self, mocked_trans, mocked_quest, mocked_info, mocked_song, mocked_and):
        """
        Test that when a song is imported and the user clicks the "yes" button, the UI goes back to the previous page
        """
        # GIVEN: A valid SongSelectForm with a mocked out QMessageBox.question() method
        mocked_trans.side_effect = lambda *args: args[1]
        mocked_quest.return_value = QtWidgets.QMessageBox.Yes
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        mocked_song_select_importer = MagicMock()
        ssform.song_select_importer = mocked_song_select_importer
        ssform.song = {'ccli_number': '1234'}

        # WHEN: The import button is clicked, and the user clicks Yes
        with patch.object(ssform, 'on_back_button_clicked') as mocked_on_back_button_clicked:
            ssform.on_import_button_clicked()

        # THEN: The on_back_button_clicked() method should have been called
        mocked_song_select_importer.save_song.assert_called_with({'ccli_number': '1234'})
        mocked_quest.assert_not_called()
        mocked_info.assert_called_once()
        mocked_on_back_button_clicked.assert_called_with(True)
        assert ssform.song is None

    @patch('openlp.plugins.songs.forms.songselectform.len')
    @patch('openlp.plugins.songs.forms.songselectform.and_')
    @patch('openlp.plugins.songs.forms.songselectform.Song')
    @patch('openlp.plugins.songs.forms.songselectform.QtWidgets.QMessageBox.information')
    @patch('openlp.plugins.songs.forms.songselectform.QtWidgets.QMessageBox.question')
    @patch('openlp.plugins.songs.forms.songselectform.translate')
    def test_on_import_duplicate_yes_clicked(self, mock_trans, mock_q, mocked_info, mock_song, mock_and, mock_len):
        """
        Test that when a duplicate song is imported and the user clicks the "yes" button, the song is imported
        """
        # GIVEN: A valid SongSelectForm with a mocked out QMessageBox.question() method
        mock_len.return_value = 1
        mock_trans.side_effect = lambda *args: args[1]
        mock_q.return_value = QtWidgets.QMessageBox.Yes
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        mocked_song_select_importer = MagicMock()
        ssform.song_select_importer = mocked_song_select_importer
        ssform.song = {'ccli_number': '1234'}

        # WHEN: The import button is clicked, and the user clicks Yes
        with patch.object(ssform, 'on_back_button_clicked') as mocked_on_back_button_clicked:
            ssform.on_import_button_clicked()

        # THEN: Should have been saved and the on_back_button_clicked() method should have been called
            mocked_song_select_importer.save_song.assert_called_with({'ccli_number': '1234'})
            mock_q.assert_called_once()
            mocked_info.assert_called_once()
            mocked_on_back_button_clicked.assert_called_once()
            assert ssform.song is None

    @patch('openlp.plugins.songs.forms.songselectform.len')
    @patch('openlp.plugins.songs.forms.songselectform.and_')
    @patch('openlp.plugins.songs.forms.songselectform.Song')
    @patch('openlp.plugins.songs.forms.songselectform.QtWidgets.QMessageBox.information')
    @patch('openlp.plugins.songs.forms.songselectform.QtWidgets.QMessageBox.question')
    @patch('openlp.plugins.songs.forms.songselectform.translate')
    def test_on_import_duplicate_no_clicked(self, mock_trans, mock_q, mocked_info, mock_song, mock_and, mock_len):
        """
        Test that when a duplicate song is imported and the user clicks the "no" button, the UI exits
        """
        # GIVEN: A valid SongSelectForm with a mocked out QMessageBox.question() method
        mock_len.return_value = 1
        mock_trans.side_effect = lambda *args: args[1]
        mock_q.return_value = QtWidgets.QMessageBox.No
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        mocked_song_select_importer = MagicMock()
        ssform.song_select_importer = mocked_song_select_importer
        ssform.song = {'ccli_number': '1234'}

        # WHEN: The import button is clicked, and the user clicks No
        with patch.object(ssform, 'on_back_button_clicked') as mocked_on_back_button_clicked:
            ssform.on_import_button_clicked()

        # THEN: Should have not been saved
            assert mocked_song_select_importer.save_song.call_count == 0
            mock_q.assert_called_once()
            mocked_info.assert_not_called()
            mocked_on_back_button_clicked.assert_not_called()
            assert ssform.song is not None

    def test_on_back_button_clicked_preview(self):
        """
        Test that when the back button is clicked on preview screen, the stacked widget is set back one page
        """
        # GIVEN: A SongSelect form, stacked widget on page 1
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        ssimporter = MagicMock()
        ssform.song_select_importer = MagicMock()
        ssform.song_select_importer.set_home_page = ssimporter
        with patch.object(ssform, 'stacked_widget') as mocked_stacked_widget:
            mocked_stacked_widget.currentIndex.return_value = 1

        # WHEN: The preview back button is clicked
            ssform.on_back_button_clicked()

        # THEN: The stacked widget should be set back one page and webpage is NOT put back to the home page
        mocked_stacked_widget.setCurrentIndex.assert_called_with(0)
        ssimporter.assert_not_called()

    def test_on_back_button_clicked_force(self):
        """
        Test that when the back button method is triggered with the force param set,
        the page should be changed
        """
        # GIVEN: A SongSelect form, stacked widget on page 1
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        ssimporter = MagicMock()
        ssform.song_select_importer = MagicMock()
        ssform.song_select_importer.set_home_page = ssimporter
        with patch.object(ssform, 'stacked_widget') as mocked_stacked_widget:
            mocked_stacked_widget.currentIndex.return_value = 1

        # WHEN: The preview back button is clicked with force param
            ssform.on_back_button_clicked(True)

        # THEN: The stacked widget should be set back one page and webpage is NOT put back to the home page
        mocked_stacked_widget.setCurrentIndex.assert_called_with(0)
        ssimporter.assert_called_once()

    def test_on_back_button_clicked(self):
        """
        Test that when the back button is clicked, the stacked widget is set to page 0
        and set to home page
        """
        # GIVEN: A SongSelect form, stacked widget on page 0
        ssform = SongSelectForm(None, MagicMock(), MagicMock())
        ssimporter = MagicMock()
        ssform.song_select_importer = MagicMock()
        ssform.song_select_importer.set_home_page = ssimporter
        with patch.object(ssform, 'stacked_widget') as mocked_stacked_widget:
            mocked_stacked_widget.currentIndex.return_value = 0

        # WHEN: The back button is clicked
            ssform.on_back_button_clicked()

        # THEN: The stacked widget should be set back one page
        mocked_stacked_widget.setCurrentIndex.assert_called_with(0)
        ssimporter.assert_called_with()

    def test_update_song_progress(self):
        """
        Test the _update_song_progress() method
        """
        # GIVEN: A SongSelect form
        ssform = SongSelectForm(None, MagicMock(), MagicMock())

        # WHEN: _update_song_progress() is called
        with patch.object(ssform, 'song_progress_bar') as mocked_song_progress_bar:
            mocked_song_progress_bar.value.return_value = 2
            ssform._update_song_progress()

        # THEN: The song progress bar should be updated
            mocked_song_progress_bar.setValue.assert_called_with(3)

    def test_on_view_button_clicked(self):
        """
        Test that view song function is run when the view button is clicked
        """
        # GIVEN: A SongSelect form
        ssform = SongSelectForm(None, MagicMock(), MagicMock())

        # WHEN: A song result is double-clicked
        with patch.object(ssform, '_view_song') as mocked_view_song:
            ssform.on_view_button_clicked()

        # THEN: The song is fetched and shown to the user
        mocked_view_song.assert_called_with()


def test_songselect_file_import():
    """
    Test that loading a SongSelect file works correctly on various files
    """
    with SongImportTestHelper('CCLIFileImport', 'cclifile') as helper:
        helper.file_import([TEST_PATH / 'TestSong.bin'],
                           helper.load_external_result_data(TEST_PATH / 'TestSong-bin.json'))
        helper.file_import([TEST_PATH / 'TestSong.txt'],
                           helper.load_external_result_data(TEST_PATH / 'TestSong-txt.json'))