qutebrowser/qutebrowser/app.py

516 lines
19 KiB
Python
Raw Normal View History

2014-02-06 14:01:23 +01:00
# Copyright 2014 Florian Bruhin (The Compiler) <mail@qutebrowser.org>
#
# This file is part of qutebrowser.
#
# qutebrowser 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.
#
# qutebrowser 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 qutebrowser. If not, see <http://www.gnu.org/licenses/>.
2014-02-17 12:23:52 +01:00
"""Initialization of qutebrowser and application-wide things."""
2014-01-30 20:42:47 +01:00
import os
2013-12-14 22:15:16 +01:00
import sys
2014-01-17 20:03:21 +01:00
import logging
2014-02-17 20:39:15 +01:00
import functools
2014-02-05 11:40:30 +01:00
import subprocess
2014-02-18 11:57:35 +01:00
import configparser
from signal import signal, SIGINT
from argparse import ArgumentParser
from base64 import b64encode
2014-04-16 15:49:56 +02:00
import qutebrowser.config.websettings as websettings
2014-02-10 22:40:21 +01:00
# Print a nice traceback on segfault -- only available on Python 3.3+, but if
# it's unavailable, it doesn't matter much.
try:
2014-02-20 15:32:46 +01:00
import faulthandler # pylint: disable=import-error
2014-02-10 22:40:21 +01:00
except ImportError:
pass
else:
faulthandler.enable()
2014-02-17 15:39:21 +01:00
# This is a really odd place to do this, but we have to do this before
# importing PyQt or it won't work.
# See https://bugreports.qt-project.org/browse/QTBUG-36099
import qutebrowser.utils.harfbuzz as harfbuzz
harfbuzz.fix()
from PyQt5.QtWidgets import QApplication, QDialog, QMessageBox
2014-04-24 06:45:38 +02:00
from PyQt5.QtCore import pyqtSlot, QTimer, QEventLoop
2014-02-17 08:56:33 +01:00
import qutebrowser
import qutebrowser.commands.utils as cmdutils
import qutebrowser.config.style as style
2014-02-23 18:07:17 +01:00
import qutebrowser.config.config as config
2014-02-21 07:18:04 +01:00
import qutebrowser.network.qutescheme as qutescheme
import qutebrowser.keyinput.modes as modes
2014-04-09 20:47:24 +02:00
import qutebrowser.utils.message as message
from qutebrowser.widgets.mainwindow import MainWindow
2014-02-10 15:01:05 +01:00
from qutebrowser.widgets.crash import CrashDialog
from qutebrowser.keyinput.commandmode import CommandKeyParser
2014-04-24 21:12:55 +02:00
from qutebrowser.keyinput.insertmode import InsertKeyParser
2014-02-24 19:11:43 +01:00
from qutebrowser.commands.parsers import CommandParser, SearchParser
2014-04-21 15:20:41 +02:00
from qutebrowser.browser.hints import HintKeyParser
2014-01-20 12:26:02 +01:00
from qutebrowser.utils.appdirs import AppDirs
2014-03-03 21:06:10 +01:00
from qutebrowser.utils.misc import dotted_getattr
2014-04-16 10:02:34 +02:00
from qutebrowser.utils.debug import set_trace # pylint: disable=unused-import
from qutebrowser.config.conftypes import ValidationError
2013-12-14 22:15:16 +01:00
2014-01-28 23:04:02 +01:00
2014-01-19 18:20:57 +01:00
class QuteBrowser(QApplication):
2014-02-07 20:21:50 +01:00
2014-01-29 15:30:19 +01:00
"""Main object for qutebrowser.
Can be used like this:
>>> app = QuteBrowser()
>>> sys.exit(app.exec_())
2014-02-07 20:21:50 +01:00
2014-02-18 16:38:13 +01:00
Attributes:
mainwindow: The MainWindow QWidget.
commandparser: The main CommandParser instance.
searchparser: The main SearchParser instance.
2014-04-21 15:20:41 +02:00
_keyparsers: A mapping from modes to keyparsers.
2014-02-18 16:38:13 +01:00
_dirs: AppDirs instance for config/cache directories.
_args: ArgumentParser instance.
_timers: List of used QTimers so they don't get GCed.
_shutting_down: True if we're currently shutting down.
_quit_status: The current quitting status.
2014-04-22 14:02:29 +02:00
_opened_urls: List of opened URLs.
2014-02-18 16:38:13 +01:00
"""
2014-01-19 18:20:57 +01:00
def __init__(self):
super().__init__(sys.argv)
2014-02-17 20:39:15 +01:00
self._quit_status = {}
2014-02-18 16:38:13 +01:00
self._timers = []
2014-04-22 14:02:29 +02:00
self._opened_urls = []
2014-02-18 16:38:13 +01:00
self._shutting_down = False
2014-02-06 10:25:22 +01:00
sys.excepthook = self._exception_hook
2014-01-28 14:44:12 +01:00
2014-02-18 16:38:13 +01:00
self._args = self._parseopts()
2014-01-29 15:30:19 +01:00
self._initlog()
2014-02-05 12:46:35 +01:00
self._initmisc()
2014-01-17 20:03:21 +01:00
2014-02-18 16:38:13 +01:00
self._dirs = AppDirs('qutebrowser')
if self._args.confdir is None:
confdir = self._dirs.user_config_dir
elif self._args.confdir == '':
confdir = None
else:
2014-02-18 16:38:13 +01:00
confdir = self._args.confdir
try:
config.init(confdir)
except ValidationError as e:
msgbox = QMessageBox(
QMessageBox.Critical,
"Error while reading config!",
"Error while reading config:\n\n{} -> {}:\n{}".format(
e.section, e.option, e))
msgbox.exec_()
# We didn't really initialize much so far, so we just quit hard.
sys.exit(1)
self.config = config.instance
2014-04-16 15:49:56 +02:00
websettings.init()
2014-01-20 12:26:02 +01:00
2014-02-24 19:11:43 +01:00
self.commandparser = CommandParser()
self.searchparser = SearchParser()
2014-04-21 15:20:41 +02:00
self._keyparsers = {
2014-04-24 21:12:55 +02:00
'normal': CommandKeyParser(self),
'hint': HintKeyParser(self),
'insert': InsertKeyParser(self),
2014-04-21 15:20:41 +02:00
}
2014-01-29 15:30:19 +01:00
self._init_cmds()
Merge completion branch Squashed commit of the following: commit 8bbf7242e7ae3168a0e8652ad643c7c1fe8670d3 Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 21:34:35 2014 +0100 foo commit cc9a4eed6e7ca9599bcc40b40f43614040711714 Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 21:33:09 2014 +0100 fixes commit b84c9207f8596dac0f47bf72cd178f8e3ee9ebb7 Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 21:27:48 2014 +0100 Make titles bold commit d2ef899f90036b05856deb178e61de5d961e1342 Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 21:18:01 2014 +0100 Make tab work properly commit ccc407093e39f1964c8e3ff2900c66b28dc7fe56 Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 16:19:57 2014 +0100 Add FIXME, remove showhide commit cbe6d0a2b0614df839c75f1815f939c11e4ae3ba Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 16:12:55 2014 +0100 Use stylesheet for highlight commit 7eeb6e79f17bc8e7ccc1ec90f474d1c2b1c01c1e Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 16:01:50 2014 +0100 Make highlighting work commit 4fed5b9f7291109fd1c669c61501861223364b7d Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 17:49:30 2014 +0100 First attempt at marking commit 27a3bda214b7fc7741b4c12d723ccdf4073c0d5e Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 17:14:48 2014 +0100 Funny broken marking commit 2798cf75dd295ace2a33295e9a64dd373b7fd669 Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 16:25:36 2014 +0100 Make completion and filtering work commit b213d9724bc9f4bb36204bd1b2c4aaf62a454e11 Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 16:07:14 2014 +0100 Implement dynamic source models commit c7951e6b4346a1702509bea6d3694cc50550c9ed Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 14:44:31 2014 +0100 Add custom filtering commit 37b100f3a1a1f5a90592e34b268c08df684b0df9 Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 14:43:57 2014 +0100 Add default roles in model commit 1610423c1dd36c0adc35743a38e2c3a81f3641ef Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 14:43:36 2014 +0100 Remove dataChanged commit 467712630b00a9ca2407903bdba977b5e66dd2bc Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 13:57:18 2014 +0100 Fix stuff commit 00bb46e0ef46b9ae61652bde3ae10d3252263e46 Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 11:51:34 2014 +0100 First attempt at a real CommandCompletionModel commit a8be022f80d7d35049d1ae3d8431702e7a407698 Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 10:25:48 2014 +0100 Adapt completion view to size commit ad6bf69a4257880ce14b0590bcb5381c718bb744 Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 09:17:48 2014 +0100 Fix stuff commit 41c1fbfa942bab0a1a0db35f6b745b691c774b22 Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 09:16:09 2014 +0100 Hide CompletionView by default commit 2ac8596dada6c7e079807bfd698377a9d3ffd0af Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 09:13:11 2014 +0100 File shuffling commit f0c85d5bee30095324be354934f3379c1c6a5291 Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 09:04:29 2014 +0100 Cleanup, don't show completionview commit 177ac97acb2d611992869f122669d95a8dadcdbb Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 07:16:37 2014 +0100 Add FIXME for eliding commit 6b234c58928e9a231ec35e5080a27d7a8affebc4 Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 07:12:55 2014 +0100 Add TextOption to TextDocument commit 195e95459b54bbdb0545cb0e4db417ab641cea28 Author: Florian Bruhin <git@the-compiler.org> Date: Mon Jan 27 07:01:41 2014 +0100 Resize QTreeView contents based on window size commit 05b854bc812f0aabf5c6dcbdfa1dfd74ea495170 Author: Florian Bruhin <git@the-compiler.org> Date: Sun Jan 26 01:19:28 2014 +0100 Split files commit ff433b561179a562494c7d3558336a917edc497f Author: Florian Bruhin <git@the-compiler.org> Date: Sun Jan 26 00:57:18 2014 +0100 cleanup commit bdb17de2fee65411b4dd1d0ad449329f0586f1e0 Author: Florian Bruhin <git@the-compiler.org> Date: Sun Jan 26 00:15:17 2014 +0100 Make text drawing work properly commit d844152074738df88ed79c2e1efc4eec4b6e5734 Author: Florian Bruhin <git@the-compiler.org> Date: Sat Jan 25 14:03:16 2014 +0100 Split painting into functions commit 585a9f6931a407117c4232ac14844d33233922f8 Author: Florian Bruhin <git@the-compiler.org> Date: Sat Jan 25 13:50:52 2014 +0100 Use QTextDocument, things are getting better commit 96fef2114e68766590ed33fa92c6f2bbac8adb11 Author: Florian Bruhin <git@the-compiler.org> Date: Sat Jan 25 13:08:58 2014 +0100 Failed try to reimplement painting commit d967285f24a7966674c8685e9e276e560e9cc066 Author: Florian Bruhin <git@the-compiler.org> Date: Sat Jan 25 11:15:07 2014 +0100 Try drawing over text commit 2687d891cc2f1ecd00cace78492cbffc745d9e5b Author: Florian Bruhin <git@the-compiler.org> Date: Fri Jan 24 23:06:44 2014 +0100 First try at implementing delegate commit c8bada0c7607df936df04b9c7796a03cc2e1a71a Author: Florian Bruhin <git@the-compiler.org> Date: Fri Jan 24 19:59:14 2014 +0100 Don't allow categories to be selected commit 6b56df5a0e2588d89d757f583070dd2ca78a6e15 Author: Florian Bruhin <git@the-compiler.org> Date: Fri Jan 24 19:53:36 2014 +0100 Add outline:0 to TreeView commit 534ba6e191e711cc7abeea78c3bf470e9eaf0bfd Author: Florian Bruhin <git@the-compiler.org> Date: Fri Jan 24 17:57:57 2014 +0100 More styling commit 065d40c24d2746f49660485700e3e6327b28810d Author: Florian Bruhin <git@the-compiler.org> Date: Fri Jan 24 17:47:40 2014 +0100 Add more commands, more styling for TreeView commit 2362d01db40bbc923a51a8bc53ffbabfd18ebea0 Author: Florian Bruhin <git@the-compiler.org> Date: Fri Jan 24 16:50:31 2014 +0100 Adjust super() commit 43ebedff6724af0f217acc14a325febd12c4ed0b Author: Florian Bruhin <git@the-compiler.org> Date: Fri Jan 24 16:50:20 2014 +0100 Style CompletionView commit 79c1010cf54bd8a3c3f4b2b4a9f97d8ae7d603fb Author: Florian Bruhin <git@the-compiler.org> Date: Fri Jan 24 13:06:12 2014 +0100 Completion refactoring Clean up imports Whitespace Make items non-editable Rename nodeToIndex() Return QVariant() instead of None Remove comments Don't inherit TreeItem from object Get rid of getValue Get rid of child() Get rid of childCount Remove appendChild Get rid of columnCount Get rid of parent() Renames Refactor if Simplify columnCount Minor fixups commit a7a6b95f56a87ef03359ec5f9e5d45a906112845 Author: Florian Bruhin <git@the-compiler.org> Date: Fri Jan 24 13:04:28 2014 +0100 Split completion out of command.py commit cd1c6419ff4034a29a1b48c4ed6ca292944f5674 Author: Florian Bruhin <git@the-compiler.org> Date: Thu Jan 23 17:53:37 2014 +0100 Fix commit 7073edea92ff8384535f5db80c01168bb8462e75 Author: Florian Bruhin <git@the-compiler.org> Date: Thu Jan 23 17:28:03 2014 +0100 More attempts commit 7dc522667ba52e9b7f20b3b66e977d23258a3ff4 Author: Florian Bruhin <git@the-compiler.org> Date: Thu Jan 23 16:56:01 2014 +0100 Get rid of parent in setupModelData commit 5df4874e14818399494f3a47f7feea8b9f86cf69 Author: Florian Bruhin <git@the-compiler.org> Date: Thu Jan 23 16:51:52 2014 +0100 First model/view experiments
2014-01-27 21:35:12 +01:00
self.mainwindow = MainWindow()
modes.init(self)
2014-04-24 21:12:55 +02:00
modes.manager.register('normal', self._keyparsers['normal'].handle)
modes.manager.register('hint', self._keyparsers['hint'].handle)
modes.manager.register('insert', self._keyparsers['insert'].handle,
passthrough=True)
modes.manager.register('command', None, passthrough=True)
self.installEventFilter(modes.manager)
self.setQuitOnLastWindowClosed(False)
2014-04-21 15:20:41 +02:00
self._connect_signals()
modes.enter("normal")
2014-01-17 20:03:21 +01:00
2014-01-19 18:20:57 +01:00
self.mainwindow.show()
2014-01-29 15:30:19 +01:00
self._python_hacks()
timer = QTimer.singleShot(0, self._process_init_args)
2014-02-18 16:38:13 +01:00
self._timers.append(timer)
def _parseopts(self):
2014-02-19 10:58:32 +01:00
"""Parse command line options.
Return:
Argument namespace from argparse.
"""
parser = ArgumentParser("usage: %(prog)s [options]")
parser.add_argument('-l', '--log', dest='loglevel',
help='Set loglevel', default='info')
parser.add_argument('-c', '--confdir', help='Set config directory '
'(empty for no config storage)')
parser.add_argument('-d', '--debug', help='Turn on debugging options.',
action='store_true')
parser.add_argument('command', nargs='*', help='Commands to execute '
'on startup.', metavar=':command')
# URLs will actually be in command
parser.add_argument('url', nargs='*', help='URLs to open on startup.')
return parser.parse_args()
def _initlog(self):
2014-04-17 17:44:27 +02:00
"""Initialisation of the logging output.
Raise:
ValueError if there was an invalid loglevel.
"""
loglevel = 'debug' if self._args.debug else self._args.loglevel
numeric_level = getattr(logging, loglevel.upper(), None)
if not isinstance(numeric_level, int):
raise ValueError('Invalid log level: {}'.format(loglevel))
logging.basicConfig(
level=numeric_level,
format='%(asctime)s [%(levelname)s] '
'[%(module)s:%(funcName)s:%(lineno)s] %(message)s',
datefmt='%Y-%m-%d %H:%M:%S')
def _initmisc(self):
"""Initialize misc things."""
if self._args.debug:
os.environ['QT_FATAL_WARNINGS'] = '1'
self.setApplicationName("qutebrowser")
self.setApplicationVersion(qutebrowser.__version__)
2014-04-09 20:47:24 +02:00
message.init()
def _init_cmds(self):
"""Initialisation of the qutebrowser commands.
2014-04-17 17:44:27 +02:00
Registers all commands and connects their signals.
"""
2014-03-03 21:06:10 +01:00
for key, cmd in sorted(cmdutils.cmd_dict.items()):
cmd.signal.connect(self.command_handler)
if cmd.instance is not None:
func = '.'.join([cmd.instance if cmd.instance else 'app',
cmd.handler.__name__])
else:
func = cmd.handler.__name__
logging.debug("Registered command: {} -> {}".format(key, func))
def _process_init_args(self):
"""Process initial positional args.
URLs to open have no prefix, commands to execute begin with a colon.
"""
# QNetworkAccessManager::createRequest will hang for over a second, so
# we make sure the GUI is refreshed here, so the start seems faster.
self.processEvents(QEventLoop.ExcludeUserInputEvents |
QEventLoop.ExcludeSocketNotifiers)
2014-02-18 16:38:13 +01:00
for e in self._args.command:
if e.startswith(':'):
logging.debug('Startup cmd {}'.format(e))
self.commandparser.run(e.lstrip(':'))
else:
logging.debug('Startup url {}'.format(e))
2014-04-22 14:02:29 +02:00
self._opened_urls.append(e)
self.mainwindow.tabs.tabopen(e)
2014-02-21 20:06:42 +01:00
if self.mainwindow.tabs.count() == 0:
logging.debug('Opening startpage')
for url in config.get('general', 'startpage'):
self.mainwindow.tabs.tabopen(url)
2014-01-29 15:30:19 +01:00
def _python_hacks(self):
"""Get around some PyQt-oddities by evil hacks.
This sets up the uncaught exception hook, quits with an appropriate
exit status, and handles Ctrl+C properly by passing control to the
Python interpreter once all 500ms.
"""
signal(SIGINT, lambda *args: self.exit(128 + SIGINT))
timer = QTimer()
timer.start(500)
timer.timeout.connect(lambda: None)
self._timers.append(timer)
2014-04-21 15:20:41 +02:00
def _connect_signals(self):
"""Connect all signals to their slots."""
2014-04-22 10:34:43 +02:00
# syntactic sugar
kp = self._keyparsers
status = self.mainwindow.status
completion = self.mainwindow.completion
tabs = self.mainwindow.tabs
cmd = self.mainwindow.status.cmd
# misc
2014-04-21 15:20:41 +02:00
self.lastWindowClosed.connect(self.shutdown)
2014-04-22 10:34:43 +02:00
tabs.quit.connect(self.shutdown)
tabs.currentChanged.connect(self.mainwindow.update_inspector)
# status bar
modes.manager.entered.connect(status.on_mode_entered)
modes.manager.left.connect(status.on_mode_left)
# FIXME what to do here?
modes.manager.key_pressed.connect(status.on_key_pressed)
2014-04-22 10:34:43 +02:00
for obj in [kp["normal"], tabs]:
obj.set_cmd_text.connect(cmd.set_cmd_text)
# commands
cmd.got_cmd.connect(self.commandparser.run)
cmd.got_search.connect(self.searchparser.search)
cmd.got_search_rev.connect(self.searchparser.search_rev)
cmd.returnPressed.connect(tabs.setFocus)
self.searchparser.do_search.connect(tabs.cur.search)
kp["normal"].keystring_updated.connect(status.keystring.setText)
# hints
kp["hint"].fire_hint.connect(tabs.cur.fire_hint)
kp["hint"].abort_hinting.connect(tabs.cur.abort_hinting)
kp["hint"].keystring_updated.connect(tabs.cur.handle_hint_key)
tabs.hint_strings_updated.connect(kp["hint"].on_hint_strings_updated)
# messages
message.bridge.error.connect(status.disp_error)
message.bridge.info.connect(status.txt.set_temptext)
message.bridge.text.connect(status.txt.set_normaltext)
# config
2014-04-21 15:20:41 +02:00
self.config.style_changed.connect(style.invalidate_caches)
2014-04-22 10:34:43 +02:00
for obj in [tabs, completion, self.mainwindow, config.cmd_history,
websettings, kp["normal"], modes.manager]:
2014-04-22 10:34:43 +02:00
self.config.changed.connect(obj.on_config_changed)
# statusbar
tabs.cur_progress.connect(status.prog.setValue)
tabs.cur_load_finished.connect(status.prog.hide)
tabs.cur_load_finished.connect(status.url.on_loading_finished)
tabs.cur_load_started.connect(status.prog.on_load_started)
tabs.cur_scroll_perc_changed.connect(status.percentage.set_perc)
tabs.cur_statusbar_message.connect(status.txt.on_statusbar_message)
tabs.cur_url_changed.connect(status.url.set_url)
tabs.cur_link_hovered.connect(status.url.set_hover_url)
# command input / completion
cmd.esc_pressed.connect(tabs.setFocus)
cmd.clear_completion_selection.connect(
completion.on_clear_completion_selection)
cmd.hide_completion.connect(completion.hide)
cmd.textChanged.connect(completion.on_cmd_text_changed)
cmd.tab_pressed.connect(completion.on_tab_pressed)
completion.change_completed_part.connect(cmd.on_change_completed_part)
2014-04-21 15:20:41 +02:00
2014-02-17 20:30:09 +01:00
def _recover_pages(self):
"""Try to recover all open pages.
Called from _exception_hook, so as forgiving as possible.
2014-02-19 10:58:32 +01:00
Return:
A list of open pages, or an empty list.
2014-02-17 20:30:09 +01:00
"""
pages = []
if self.mainwindow is None:
return pages
if self.mainwindow.tabs is None:
return pages
for tabidx in range(self.mainwindow.tabs.count()):
try:
url = self.mainwindow.tabs.widget(tabidx).url().toString()
if url:
pages.append(url)
except Exception: # pylint: disable=broad-except
pass
return pages
def _save_geometry(self):
"""Save the window geometry to the state config."""
geom = b64encode(bytes(self.mainwindow.saveGeometry())).decode('ASCII')
try:
config.state.add_section('geometry')
except configparser.DuplicateSectionError:
pass
config.state['geometry']['mainwindow'] = geom
2014-01-30 20:42:47 +01:00
def _exception_hook(self, exctype, excvalue, tb):
2014-01-29 15:30:19 +01:00
"""Handle uncaught python exceptions.
It'll try very hard to write all open tabs to a file, and then exit
gracefully.
"""
2014-01-28 19:52:09 +01:00
# pylint: disable=broad-except
2014-01-30 20:42:47 +01:00
exc = (exctype, excvalue, tb)
2014-02-06 10:25:22 +01:00
sys.__excepthook__(*exc)
2014-01-30 20:42:47 +01:00
if not issubclass(exctype, Exception):
2014-02-17 20:30:09 +01:00
# probably a KeyboardInterrupt
try:
self.shutdown()
return
except Exception:
self.quit()
2014-02-17 20:39:15 +01:00
self._quit_status['crash'] = False
self._quit_status['shutdown'] = False
2014-01-28 14:44:12 +01:00
try:
2014-02-17 20:30:09 +01:00
pages = self._recover_pages()
2014-01-28 14:44:12 +01:00
except Exception:
2014-02-17 20:30:09 +01:00
pages = []
2014-01-30 20:42:47 +01:00
try:
history = self.mainwindow.status.cmd.history[-5:]
except Exception:
history = []
# Try to shutdown gracefully
try:
self.shutdown(do_quit=False)
except Exception:
pass
try:
self.lastWindowClosed.disconnect(self.shutdown)
except TypeError:
logging.exception("Preventing shutdown failed.")
2014-02-05 11:40:30 +01:00
QApplication.closeAllWindows()
2014-01-30 20:42:47 +01:00
dlg = CrashDialog(pages, history, exc)
ret = dlg.exec_()
if ret == QDialog.Accepted: # restore
os.environ['PYTHONPATH'] = os.pathsep.join(sys.path)
2014-04-22 14:02:29 +02:00
argv = sys.argv[:]
for page in self._opened_urls:
try:
argv.remove(page)
except ValueError:
pass
argv = [sys.executable] + argv + pages
2014-01-30 20:42:47 +01:00
logging.debug('Running {} with args {}'.format(sys.executable,
argv))
2014-02-05 11:40:30 +01:00
subprocess.Popen(argv)
2014-02-17 20:39:15 +01:00
self._maybe_quit('crash')
def _maybe_quit(self, sender):
"""Maybe quit qutebrowser.
This only quits if both the CrashDialog was ready to quit AND the
shutdown is complete.
2014-02-17 22:21:11 +01:00
2014-02-19 10:58:32 +01:00
Args:
The sender of the quit signal (string)
2014-02-17 20:39:15 +01:00
"""
self._quit_status[sender] = True
2014-02-17 22:21:11 +01:00
logging.debug("maybe_quit called from {}, quit status {}".format(
sender, self._quit_status))
2014-02-17 20:39:15 +01:00
if all(self._quit_status.values()):
2014-02-18 13:05:42 +01:00
logging.debug("maybe_quit quitting.")
self.quit()
2014-01-28 14:44:12 +01:00
@cmdutils.register(instance='', maxsplit=0)
2014-01-19 23:54:22 +01:00
def pyeval(self, s):
2014-01-29 15:30:19 +01:00
"""Evaluate a python string and display the results as a webpage.
:pyeval command handler.
2014-02-07 20:21:50 +01:00
2014-02-19 10:58:32 +01:00
Args:
s: The string to evaluate.
2014-01-29 15:30:19 +01:00
"""
2014-01-19 23:54:22 +01:00
try:
r = eval(s)
out = repr(r)
2014-01-28 23:04:02 +01:00
except Exception as e: # pylint: disable=broad-except
2014-01-19 23:54:22 +01:00
out = ': '.join([e.__class__.__name__, str(e)])
2014-02-21 07:18:04 +01:00
qutescheme.pyeval_output = out
self.mainwindow.tabs.cur.openurl('qute:pyeval')
2014-01-30 14:58:32 +01:00
2014-03-03 21:06:10 +01:00
@cmdutils.register(instance='', hide=True)
2014-01-30 14:58:32 +01:00
def crash(self):
"""Crash for debugging purposes.
:crash command handler.
2014-02-07 20:21:50 +01:00
2014-02-19 10:58:32 +01:00
Raises:
Always raises Exception.
2014-01-30 14:58:32 +01:00
"""
2014-02-19 11:10:26 +01:00
raise Exception("Forced crash")
@pyqtSlot()
2014-03-03 21:11:02 +01:00
@cmdutils.register(instance='', name=['quit', 'q'], nargs=0)
def shutdown(self, do_quit=True):
"""Try to shutdown everything cleanly.
For some reason lastWindowClosing sometimes seem to get emitted twice,
so we make sure we only run once here.
2014-02-19 10:58:32 +01:00
Args:
do_quit: Whether to quit after shutting down.
"""
if self._shutting_down:
return
self._shutting_down = True
logging.debug("Shutting down... (do_quit={})".format(do_quit))
if config.get('general', 'autosave'):
2014-04-15 17:28:05 +02:00
try:
config.instance.save()
2014-04-15 17:28:05 +02:00
except AttributeError:
logging.exception("Could not save config.")
2014-04-15 18:02:07 +02:00
try:
config.cmd_history.save()
except AttributeError:
logging.exception("Could not save command history.")
try:
self._save_geometry()
config.state.save()
except AttributeError:
logging.exception("Could not save window geometry.")
try:
if do_quit:
self.mainwindow.tabs.shutdown_complete.connect(
self.on_tab_shutdown_complete)
else:
self.mainwindow.tabs.shutdown_complete.connect(
functools.partial(self._maybe_quit, 'shutdown'))
self.mainwindow.tabs.shutdown()
except AttributeError: # mainwindow or tabs could still be None
logging.exception("No mainwindow/tabs to shut down.")
if do_quit:
self.quit()
@pyqtSlot()
def on_tab_shutdown_complete(self):
"""Quit application after a shutdown.
Gets called when all tabs finished shutting down after shutdown().
"""
logging.debug("Shutdown complete, quitting.")
self.quit()
2014-03-03 21:06:10 +01:00
@pyqtSlot(tuple)
def command_handler(self, tpl):
"""Handle commands which need an instance..
Args:
tpl: An (instance, func, count, args) tuple.
instance: How to get the current instance of the target object
from app.py, as a dotted string, e.g.
'mainwindow.tabs.cur'.
func: The function name to be called (as string).
count: The count given to the command, or None.
args: A list of arguments given to the command.
"""
(instance, func, count, args) = tpl
if instance == '':
obj = self
else:
obj = dotted_getattr(self, instance)
handler = getattr(obj, func)
if count is not None:
handler(*args, count=count)
else:
handler(*args)