Source code for taurus.qt.qtgui.application.taurusapplication

#!/usr/bin/env python

# ###########################################################################
#
# This file is part of Taurus
#
# http://taurus-scada.org
#
# Copyright 2011 CELLS / ALBA Synchrotron, Bellaterra, Spain
#
# Taurus is free software: you can redistribute it and/or modify
# it under the terms of the GNU Lesser General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# Taurus 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 Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with Taurus.  If not, see <http://www.gnu.org/licenses/>.
#
# ###########################################################################

"""This module provides the base
:class:`taurus.qt.qtgui.application.TaurusApplication` class.
"""

import os
import sys
import logging
import threading
import signal

from taurus.external.qt import Qt

from taurus.core.util.log import LogExceptHook, Logger
from taurus.taurusstatecontroller import application
from taurus import tauruscustomsettings


__docformat__ = "restructuredtext"


class STD(Logger):

    FlushWaterMark = 1000

    def __init__(
        self, name="", parent=None, format=None, std=None, pure_text=True
    ):
        """The STD Logger constructor

        :param name: the logger name (default is empty string)
        :type name: str
        :param parent: the parent logger or None if no parent exists (default
            is None)
        :type parent: Logger
        :param format: the log message format or None to use the default log
            format (default is None)
        :type format: str
        :param std: std to forward write
        :param pure_text: if True, writes the 'message' parameter of the log
            message in a separate line preserving the indentation
        """
        Logger.__init__(self, name=name, parent=parent, format=format)
        self.buffer = ""
        self.log_obj.propagate = False
        self.std = std

    # Trying to mimic stderr
    @property
    def errors(self):
        return self.std.errors

    def addLogHandler(self, handler):
        """When called, set to use a private handler and DON'T send messages
        to parent loggers (basically will act as an independent logging system
        by itself)

        :param handler: new handler
        """
        Logger.addLogHandler(self, handler)
        self.log_obj.propagate = not len(self.log_handlers)

    def write(self, msg):
        try:
            self.buffer += msg
            # while there is no new line, just accumulate the buffer
            msgl = len(msg)
            if msgl > 0 and (
                msg[-1] == "\n"
                or msg.index("\n") >= 0
                or msgl >= self.FlushWaterMark
            ):
                self.flush()
        except ValueError:
            pass
        finally:
            if self.std is not None:
                try:
                    self.std.write(msg)
                except Exception:
                    pass
            pass

    def flush(self):
        try:
            buff = self.buffer
            if buff is None or len(buff) == 0:
                return
            # take the '\n' because the output is a list of strings, each to be
            # interpreted as a separate line in the client
            if buff[-1] == "\n":
                buff = buff[:-1]
            if self.log_handlers:
                self.log(Logger.Console, "\n" + buff)
            self.buffer = ""
        finally:
            if self.std is not None:
                try:
                    self.std.flush()
                except Exception:
                    pass
            pass


[docs] class TaurusApplication(Qt.QApplication, Logger): """A QApplication that performs some taurus-specific initializations and (optionally but deprecated) also parses the command line for taurus options. The optional keyword parameters: - app_name: (str) application name - app_version: (str) application version - org_name: (str) organization name - org_domain: (str) organization domain - cmd_line_parser (None [or DEPRECATED :class:`optparse.OptionParser`]) If cmd_line_parser is explicitly set to None, no parsing will be done at all. If a :class:`optparse.OptionParser` instance is passed as cmd_line_parser (deprecated), it will be used for parsing the command line arguments. If cmd_line_parser is not explicitly passed, the behaviour will depend on the value of tauruscustomsettings.IMPLICIT_OPTPARSE (which by default is False, indicating that no parsing is done). Simple example:: import sys from taurus.qt.qtgui.application import TaurusApplication import taurus.qt.qtgui.display app = TaurusApplication(cmd_line_parser=None) w = taurus.qt.qtgui.display.TaurusLabel() w.model = 'sys/tg_test/1/double_scalar' w.show() sys.exit(app.exec_()) """ def __init__(self, *args, **kwargs): """The constructor. Parameters are the same as QApplication plus a keyword parameter: 'cmd_line_parser' which should be None or an instance of :class:`optparse.OptionParser`. If cmd_line_parser is None no command line parsing will be done () """ application.application_starting() # lock to safely get singleton elements (like IPython taurus # console app) self._lock = threading.Lock() if len(args) == 0: args = (getattr(sys, "argv", []),) app_name = kwargs.pop("app_name", None) app_version = kwargs.pop("app_version", None) org_name = kwargs.pop("org_name", None) org_domain = kwargs.pop("org_domain", None) if getattr(tauruscustomsettings, "IMPLICIT_OPTPARSE", True): import optparse default_parser = optparse.OptionParser() else: default_parser = None parser = kwargs.pop("cmd_line_parser", default_parser) ####################################################################### # Workaround for XInitThreads-related crash. # Sometimes (e.g. running `taurusgui example01`) one gets: # [xcb] Unknown request in queue while dequeuing # [xcb] Most likely this is a multi-threaded client and XInitThreads # has not been called # [xcb] Aborting, sorry about that. # # According to http://stackoverflow.com/a/31967769 , it is fixed by: try: Qt.QCoreApplication.setAttribute(Qt.Qt.AA_X11InitThreads) except AttributeError: pass # Qt < 4.8 does not define AA_X11InitThreads ###################################################################### try: Qt.QApplication.__init__(self, *args, **kwargs) except TypeError: Qt.QApplication.__init__(self, *args) Logger.__init__(self) self._out = None self._err = None if app_name is not None: self.setApplicationName(app_name) if app_version is not None: self.setApplicationVersion(app_version) if org_name is not None: self.setOrganizationName(org_name) if org_domain is not None: self.setOrganizationDomain(org_domain) if parser is None: p, opt, args = None, None, None else: try: if parser.version is None and app_version: # if the parser does not contain version information # but we have an application version, add the # --version capability v = app_version if app_name: v = app_name + " " + app_version parser.version = v parser._add_version_option() import taurus.core.util.argparse p, opt, args = taurus.core.util.argparse.init_taurus_args( parser=parser, args=args[0][1:] ) except Exception: self.error( "Error while using the given cmd_line_parser. \n" + "hint: it must be either None or an " + "optparse.OptionParser instance" ) raise self._cmd_line_parser = p self._cmd_line_options = opt self._cmd_line_args = args self.__registerQtLogger() self.__registerExtensions() self.__redirect_std() signal.signal(signal.SIGINT, self.exit_handler) def __registerQtLogger(self): import taurus.qt.qtcore.util taurus.qt.qtcore.util.initTaurusQtLogger() def __registerExtensions(self): """Registers taurus Qt extensions""" try: import sardana.taurus.qt.qtcore.tango.sardana sardana.taurus.qt.qtcore.tango.sardana.registerExtensions() except Exception: self.debug("Failed to load sardana extensions", exc_info=1) try: import taurus.core.tango.img taurus.core.tango.img.registerExtensions() except Exception: self.debug("Failed to load image extensions", exc_info=1) def __redirect_std(self): """Internal method to redirect stdout and stderr to log messages""" Logger.addLevelName(Logger.Critical + 10, "CONSOLE") # only redirect if display hook has not been set (IPython does it) if sys.displayhook == sys.__displayhook__: self._out = STD(name="OUT", std=sys.stdout) sys.stdout = self._out self._err = STD(name="ERR", std=sys.stderr) sys.stderr = self._err def __buildLogFileName(self, prefix=None, name=None): if prefix is None: prefix = os.path.expanduser("~/tmp") appName = str(self.applicationName()) if not appName: appName = os.path.splitext(os.path.basename(sys.argv[0]))[0] dirName = os.path.join(prefix, appName) if not os.path.isdir(dirName): os.makedirs(dirName) if name is None: name = appName + ".log" fileName = os.path.join(dirName, name) return fileName
[docs] def get_command_line_parser(self): """Returns the :class:`optparse.OptionParser` used to parse the command line parameters. :return: the parser used in the command line :rtype: :class:`optparse.OptionParser` """ # TODO: issue a warning if this is uninitialized (cmd_line_parser=None) return self._cmd_line_parser
[docs] def get_command_line_options(self): """Returns the :class:`optparse.Option` that resulted from parsing the command line parameters. :return: the command line options :rtype: :class:`optparse.Option` """ # TODO: issue a warning if this is uninitialized (cmd_line_parser=None) return self._cmd_line_options
[docs] def get_command_line_args(self): """Returns the list of arguments that resulted from parsing the command line parameters. :return: the command line arguments :rtype: list of strings """ # TODO: issue a warning if this is uninitialized (cmd_line_parser=None) return self._cmd_line_args
[docs] def setTaurusStyle(self, styleName): """Sets taurus application style to the given style name :param styleName: the new style name to be applied :type styleName: str """ import taurus.qt.qtgui.style taurus.qt.qtgui.style.setTaurusStyle(styleName)
[docs] def basicConfig( self, log_file_name=None, maxBytes=1e7, backupCount=5, with_gui_exc_handler=True, ): # prepare exception handler to report exceptions as error log messages # and to taurus message box (if with_gui is set) hook_to = None if with_gui_exc_handler: import taurus.qt.qtgui.dialog hook_to = taurus.qt.qtgui.dialog.TaurusExceptHookMessageBox() sys.excepthook = LogExceptHook(hook_to=hook_to) # create a file log handler try: if log_file_name is None: log_file_name = self.__buildLogFileName() f_h = logging.handlers.RotatingFileHandler( log_file_name, maxBytes=maxBytes, backupCount=backupCount ) Logger.addRootLogHandler(f_h) if self._out is not None: self._out.std = sys.__stdout__ self._out.addLogHandler(f_h) if self._out is not None: self._err.std = sys.__stderr__ self._err.addLogHandler(f_h) self.info("Logs will be saved in %s", log_file_name) except Exception: self.warning( "'%s' could not be created. Logs will not be stored", log_file_name, ) self.debug("Error description", exc_info=1)
[docs] def exit_handler(self, sig, frame): self.info("Ctrl+C pressed. Exitting...") for widget in Qt.QApplication.topLevelWidgets(): # Send close event to TaurusMainWindow # to manage SAVE_SETTINGS_ON_CLOSE if isinstance(widget, Qt.QMainWindow): self.debug(f"Closing widget: {widget.windowTitle()}") widget.close() # Send QCloseEvent to widget Qt.QApplication.quit()
[docs] @staticmethod def exec_(*args, **kwargs): # TODO: substitute this ugly hack by some more general mechanism try: application.application_started() ret = Qt.QApplication.exec_(*args, **kwargs) except TypeError: ret = Qt.QApplication.exec_(*args) from taurus._taurushelper import _DEPRECATION_COUNT # noqa from taurus import info info("\n*********************\n%s", _DEPRECATION_COUNT.pretty()) return ret