# -*- coding: utf-8 -*- ########################################################################## # OpenLP - Open Source Lyrics Projection # # ---------------------------------------------------------------------- # # Copyright (c) 2008-2022 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 . # ########################################################################## """ The :mod:`core` module provides all core application functions All the core functions of the OpenLP application including the GUI, settings, logging and a plugin framework are contained within the openlp.core module. """ import argparse import logging import os import sys import time from datetime import datetime from pathlib import Path from shutil import copytree from traceback import format_exception from PyQt5 import QtCore, QtGui, QtWebEngineWidgets, QtWidgets # noqa from openlp.core.api.deploy import check_for_remote_update from openlp.core.common import is_macosx, is_win from openlp.core.common.applocation import AppLocation from openlp.core.common.i18n import LanguageManager, UiStrings, translate from openlp.core.common.mixins import LogMixin from openlp.core.common.path import create_paths from openlp.core.common.registry import Registry from openlp.core.common.settings import Settings from openlp.core.display.screens import ScreenList from openlp.core.loader import loader from openlp.core.resources import qInitResources from openlp.core.server import Server from openlp.core.state import State from openlp.core.ui.exceptionform import ExceptionForm from openlp.core.ui.firsttimeform import FirstTimeForm from openlp.core.ui.firsttimelanguageform import FirstTimeLanguageForm from openlp.core.ui.mainwindow import MainWindow from openlp.core.ui.splashscreen import SplashScreen from openlp.core.ui.style import get_application_stylesheet, set_default_theme from openlp.core.version import check_for_update, get_version __all__ = ['OpenLP', 'main'] log = logging.getLogger() class OpenLP(QtCore.QObject, LogMixin): """ The core worker class. This class that holds the whole system together. """ args = [] worker_threads = {} def exec(self): """ Override exec method to allow the shared memory to be released on exit """ self.is_event_loop_active = True result = QtWidgets.QApplication.exec() if hasattr(self, 'server'): self.server.close_server() return result def run(self, args, app): """ Run the OpenLP application. :param args: Some Args """ self.is_event_loop_active = False # On Windows, the args passed into the constructor are ignored. Not very handy, so set the ones we want to use. # On Linux and FreeBSD, in order to set the WM_CLASS property for X11, we pass "OpenLP" in as a command line # argument. This interferes with files being passed in as command line arguments, so we remove it from the list. if 'OpenLP' in args: args.remove('OpenLP') self.args.extend(args) # Decide how many screens we have and their size screens = ScreenList.create(app) # First time checks in settings has_run_wizard = self.settings.value('core/has run wizard') if not has_run_wizard: ftw = FirstTimeForm() ftw.initialize(screens) if ftw.exec() == QtWidgets.QDialog.Accepted: self.settings.setValue('core/has run wizard', True) else: QtCore.QCoreApplication.exit() sys.exit() can_show_splash = self.settings.value('core/show splash') if can_show_splash: self.splash = SplashScreen() self.splash.show() # make sure Qt really display the splash screen QtWidgets.QApplication.processEvents() # Check if OpenLP has been upgrade and if a backup of data should be created self.backup_on_upgrade(has_run_wizard, can_show_splash) # start the main app window loader() # Set the darkmode based on theme set_default_theme(app) self.main_window = MainWindow() self.main_window.installEventFilter(self.main_window) # Correct stylesheet bugs application_stylesheet = get_application_stylesheet() if application_stylesheet: self.main_window.setStyleSheet(application_stylesheet) Registry().execute('bootstrap_initialise') State().flush_preconditions() Registry().execute('bootstrap_post_set_up') Registry().initialise = False self.main_window.show() if can_show_splash: # now kill the splashscreen log.debug('Splashscreen closing') self.splash.close() log.debug('Splashscreen closed') # make sure Qt really display the splash screen QtWidgets.QApplication.processEvents() self.main_window.repaint() QtWidgets.QApplication.processEvents() if not has_run_wizard: self.main_window.first_time() if self.settings.value('core/update check'): check_for_update(self.main_window) if self.settings.value('api/update check'): check_for_remote_update(self.main_window) self.main_window.is_display_blank() Registry().execute('bootstrap_completion') return self.exec() @staticmethod def is_already_running(): """ Tell the user there is a 2nd instance running. """ QtWidgets.QMessageBox.critical(None, UiStrings().Error, UiStrings().OpenLPStart, QtWidgets.QMessageBox.StandardButtons(QtWidgets.QMessageBox.Ok)) def is_data_path_missing(self): """ Check if the data folder path exists. """ data_folder_path = AppLocation.get_data_path() if not data_folder_path.exists(): log.critical('Database was not found in: %s', data_folder_path) status = QtWidgets.QMessageBox.critical( None, translate('OpenLP', 'Data Directory Error'), translate('OpenLP', 'OpenLP data folder was not found in:\n\n{path}\n\nThe location of the data folder ' 'was previously changed from the OpenLP\'s default location. If the data was ' 'stored on removable device, that device needs to be made available.\n\nYou may ' 'reset the data location back to the default location, or you can try to make the ' 'current location available.\n\nDo you want to reset to the default data location? ' 'If not, OpenLP will be closed so you can try to fix the problem.') .format(path=data_folder_path), QtWidgets.QMessageBox.StandardButtons(QtWidgets.QMessageBox.Yes | QtWidgets.QMessageBox.No), QtWidgets.QMessageBox.No) if status == QtWidgets.QMessageBox.No: # If answer was "No", return "True", it will shutdown OpenLP in def main log.info('User requested termination') return True # If answer was "Yes", remove the custom data path thus resetting the default location. self.settings.remove('advanced/data path') log.info('Database location has been reset to the default settings.') return False def hook_exception(self, exc_type, value, traceback): """ Add an exception hook so that any uncaught exceptions are displayed in this window rather than somewhere where users cannot see it and cannot report when we encounter these problems. :param exc_type: The class of exception. :param value: The actual exception object. :param traceback: A traceback object with the details of where the exception occurred. """ # We can't log.exception here because the last exception no longer exists, we're actually busy handling it. log.critical(''.join(format_exception(exc_type, value, traceback))) if not hasattr(self, 'exception_form'): self.exception_form = ExceptionForm() self.exception_form.exception_text_edit.setPlainText(''.join(format_exception(exc_type, value, traceback))) self.set_normal_cursor() is_splash_visible = False if hasattr(self, 'splash') and self.splash.isVisible(): is_splash_visible = True self.splash.hide() self.exception_form.exec() if is_splash_visible: self.splash.show() def backup_on_upgrade(self, has_run_wizard, can_show_splash): """ Check if OpenLP has been upgraded, and ask if a backup of data should be made :param has_run_wizard: OpenLP has been run before :param can_show_splash: Should OpenLP show the splash screen """ data_version = self.settings.value('core/application version') openlp_version = get_version()['version'] # New installation, no need to create backup if not has_run_wizard: self.settings.setValue('core/application version', openlp_version) # If data_version is different from the current version ask if we should backup the data folder elif data_version != openlp_version: if can_show_splash and self.splash.isVisible(): self.splash.hide() if QtWidgets.QMessageBox.question(None, translate('OpenLP', 'Backup'), translate('OpenLP', 'OpenLP has been upgraded, do you want to create\n' 'a backup of the old data folder?'), defaultButton=QtWidgets.QMessageBox.Yes) == QtWidgets.QMessageBox.Yes: # Create copy of data folder data_folder_path = AppLocation.get_data_path() timestamp = time.strftime("%Y%m%d-%H%M%S") data_folder_backup_path = data_folder_path.with_name(data_folder_path.name + '-' + timestamp) try: copytree(data_folder_path, data_folder_backup_path) except OSError: QtWidgets.QMessageBox.warning(None, translate('OpenLP', 'Backup'), translate('OpenLP', 'Backup of the data folder failed!')) return message = translate('OpenLP', 'A backup of the data folder has been created at:\n\n' '{text}').format(text=data_folder_backup_path) QtWidgets.QMessageBox.information(None, translate('OpenLP', 'Backup'), message) # Update the version in the settings self.settings.setValue('core/application version', openlp_version) if can_show_splash: self.splash.show() @staticmethod def process_events(): """ Wrapper to make ProcessEvents visible and named correctly """ QtWidgets.QApplication.processEvents() @staticmethod def set_busy_cursor(): """ Sets the Busy Cursor for the Application """ QtWidgets.QApplication.setOverrideCursor(QtCore.Qt.BusyCursor) QtWidgets.QApplication.processEvents() @staticmethod def set_normal_cursor(): """ Sets the Normal Cursor for the Application """ QtWidgets.QApplication.restoreOverrideCursor() QtWidgets.QApplication.processEvents() def parse_options(): """ Parse the command line arguments :return: An :object:`argparse.Namespace` insatnce containing the parsed args. :rtype: argparse.Namespace """ # Set up command line options. parser = argparse.ArgumentParser(prog='openlp') parser.add_argument('-e', '--no-error-form', dest='no_error_form', action='store_true', help='Disable the error notification form.') parser.add_argument('-l', '--log-level', dest='loglevel', default='warning', metavar='LEVEL', help='Set logging to LEVEL level. Valid values are "debug", "info", "warning".') parser.add_argument('-p', '--portable', dest='portable', action='store_true', help='Specify if this should be run as a portable app, ') parser.add_argument('-pp', '--portable-path', dest='portablepath', default=None, help='Specify the path of the portable data, defaults to "{dir_name}".'.format( dir_name=os.path.join('', '..', '..'))) parser.add_argument('-w', '--no-web-server', dest='no_web_server', action='store_true', help='Turn off the Web and Socket Server ') parser.add_argument('rargs', nargs='*', default=[]) # Parse command line options and deal with them. return parser.parse_args() def set_up_logging(log_path): """ Setup our logging using log_path :param Path log_path: The file to save the log to. :rtype: None """ create_paths(log_path, do_not_log=True) file_path = log_path / 'openlp.log' logfile = logging.FileHandler(file_path, 'w', encoding='UTF-8') logfile.setFormatter(logging.Formatter('%(asctime)s %(threadName)s %(name)-55s %(levelname)-8s %(message)s')) log.addHandler(logfile) if log.isEnabledFor(logging.DEBUG): print('Logging to: {name}'.format(name=file_path)) def main(): """ The main function which parses command line options and then runs """ log.debug('Entering function - main') args = parse_options() qt_args = ['--disable-web-security'] # qt_args = [] if args and args.loglevel.lower() in ['d', 'debug']: log.setLevel(logging.DEBUG) elif args and args.loglevel.lower() in ['w', 'warning']: log.setLevel(logging.WARNING) else: log.setLevel(logging.INFO) # Throw the rest of the arguments at Qt, just in case. qt_args.extend(args.rargs) # Bug #1018855: Set the WM_CLASS property in X11 if not is_win() and not is_macosx(): qt_args.append('OpenLP') elif is_win(): # support dark mode on windows 10. This makes the titlebar dark, the rest is setup later # by calling set_windows_darkmode qt_args.extend(['-platform', 'windows:darkmode=1']) # Initialise the resources qInitResources() # Now create and actually run the application. QtWidgets.QApplication.setAttribute(QtCore.Qt.AA_EnableHighDpiScaling) application = QtWidgets.QApplication(qt_args) application.setOrganizationName('OpenLP') application.setOrganizationDomain('openlp.org') application.setAttribute(QtCore.Qt.AA_UseHighDpiPixmaps, True) application.setAttribute(QtCore.Qt.AA_DontCreateNativeWidgetSiblings, True) if args.portable: application.setApplicationName('OpenLPPortable') Settings.setDefaultFormat(Settings.IniFormat) # Get location OpenLPPortable.ini if args.portablepath: if os.path.isabs(args.portablepath): portable_path = Path(args.portablepath) else: portable_path = AppLocation.get_directory(AppLocation.AppDir) / '..' / args.portablepath else: portable_path = AppLocation.get_directory(AppLocation.AppDir) / '..' / '..' portable_path = portable_path.resolve() data_path = portable_path / 'Data' set_up_logging(portable_path / 'Other') log.info('Running portable') portable_settings_path = data_path / 'OpenLP.ini' # Make this our settings file log.info('INI file: {name}'.format(name=portable_settings_path)) Settings.set_filename(portable_settings_path) portable_settings = Settings() # Set our data path log.info('Data path: {name}'.format(name=data_path)) # Point to our data path portable_settings.setValue('advanced/data path', data_path) portable_settings.setValue('advanced/is portable', True) portable_settings.sync() else: application.setApplicationName('OpenLP') set_up_logging(AppLocation.get_directory(AppLocation.CacheDir)) # Set the libvlc environment variable if we're frozen if getattr(sys, 'frozen', False) and is_win(): # Path to libvlc and the plugins os.environ['PYTHON_VLC_LIB_PATH'] = str(AppLocation.get_directory(AppLocation.AppDir) / 'vlc' / 'libvlc.dll') os.environ['PYTHON_VLC_MODULE_PATH'] = str(AppLocation.get_directory(AppLocation.AppDir) / 'vlc') os.environ['PATH'] += ';' + str(AppLocation.get_directory(AppLocation.AppDir) / 'vlc') log.debug('VLC Path: {}'.format(os.environ['PYTHON_VLC_LIB_PATH'])) app = OpenLP() # Initialise the Registry Registry.create() settings = Settings() Registry().register('settings', settings) # Need settings object for the threads. settings_thread = Settings() Registry().register('settings_thread', settings_thread) Registry().register('application-qt', application) Registry().register('application', app) Registry().set_flag('no_web_server', args.no_web_server) # Upgrade settings. app.settings = settings application.setApplicationVersion(get_version()['version']) # Check if an instance of OpenLP is already running. Quit if there is a running instance and the user only wants one server = Server() if server.is_another_instance_running(): app.is_already_running() server.post_to_server(qt_args) sys.exit() else: server.start_server() app.server = server # If the custom data path is missing and the user wants to restore the data path, quit OpenLP. if app.is_data_path_missing(): server.close_server() sys.exit() if settings.can_upgrade(): now = datetime.now() # Only back up if OpenLP has previously run. if settings.value('core/has run wizard'): back_up_path = AppLocation.get_data_path() / (now.strftime('%Y-%m-%d %H-%M') + '.conf') log.info('Settings about to be upgraded. Existing settings are being backed up to {back_up_path}' .format(back_up_path=back_up_path)) QtWidgets.QMessageBox.information( None, translate('OpenLP', 'Settings Upgrade'), translate('OpenLP', 'Your settings are about to be upgraded. A backup will be created at ' '{back_up_path}').format(back_up_path=back_up_path)) try: settings.export(back_up_path) except OSError: QtWidgets.QMessageBox.warning( None, translate('OpenLP', 'Settings Upgrade'), translate('OpenLP', 'Settings back up failed.\n\nContinuing to upgrade.')) settings.upgrade_settings() # First time checks in settings if not settings.value('core/has run wizard'): if not FirstTimeLanguageForm().exec(): # if cancel then stop processing server.close_server() sys.exit() # i18n Set Language language = LanguageManager.get_language() translators = LanguageManager.get_translators(language) for translator in translators: if not translator.isEmpty(): application.installTranslator(translator) if not translators: log.debug('Could not find translators.') if args and not args.no_error_form: sys.excepthook = app.hook_exception sys.exit(app.run(qt_args, application))