This commit is contained in:
thuck 2016-11-17 07:46:03 +01:00
commit bcb0010fcb
69 changed files with 543 additions and 717 deletions

View File

@ -45,6 +45,7 @@ matrix:
- os: osx - os: osx
env: TESTENV=py35 OSX=elcapitan env: TESTENV=py35 OSX=elcapitan
osx_image: xcode7.3 osx_image: xcode7.3
fast_finish: true
cache: cache:
directories: directories:

View File

@ -50,6 +50,8 @@ Added
- New `cast` userscript to show a video on a Google Chromecast - New `cast` userscript to show a video on a Google Chromecast
- New `:run-with-count` command which replaces the (undocumented) `:count:command` syntax. - New `:run-with-count` command which replaces the (undocumented) `:count:command` syntax.
- New `:record-macro` (`q`) and `:run-macro` (`@`) commands for keyboard macros. - New `:record-macro` (`q`) and `:run-macro` (`@`) commands for keyboard macros.
- New `ui -> hide-scrollbar` setting to hide the scrollbar independently of the
`user-stylesheet` setting.
Changed Changed
~~~~~~~ ~~~~~~~
@ -143,6 +145,8 @@ Changed
- Various functionality now works when javascript is disabled with QtWebKit - Various functionality now works when javascript is disabled with QtWebKit
- Various commands/settings taking `left`/`right`/`previous` arguments now take - Various commands/settings taking `left`/`right`/`previous` arguments now take
`prev`/`next`/`last-used` to remove ambiguity. `prev`/`next`/`last-used` to remove ambiguity.
- The `ui -> user-stylesheet` setting now only takes filenames, not CSS snippets
- `ui -> window-title-format` now has a new `{backend} ` replacement
Deprecated Deprecated
~~~~~~~~~~ ~~~~~~~~~~
@ -168,6 +172,8 @@ Removed
thus removed. thus removed.
- All `--qt-*` arguments got replaced by `--qt-arg` and `--qt-flag` and thus - All `--qt-*` arguments got replaced by `--qt-arg` and `--qt-flag` and thus
removed. removed.
- The `-c`/`--confdir`, `--datadir` and `--cachedir` arguments got removed, as
`--basedir` should be sufficient.
Fixed Fixed
~~~~~ ~~~~~

View File

@ -44,7 +44,8 @@
|<<ui-confirm-quit,confirm-quit>>|Whether to confirm quitting the application. |<<ui-confirm-quit,confirm-quit>>|Whether to confirm quitting the application.
|<<ui-zoom-text-only,zoom-text-only>>|Whether the zoom factor on a frame applies only to the text or to all content. |<<ui-zoom-text-only,zoom-text-only>>|Whether the zoom factor on a frame applies only to the text or to all content.
|<<ui-frame-flattening,frame-flattening>>|Whether to expand each subframe to its contents. |<<ui-frame-flattening,frame-flattening>>|Whether to expand each subframe to its contents.
|<<ui-user-stylesheet,user-stylesheet>>|User stylesheet to use (absolute filename, filename relative to the config directory or CSS string). Will expand environment variables. |<<ui-user-stylesheet,user-stylesheet>>|User stylesheet to use (absolute filename or filename relative to the config directory). Will expand environment variables.
|<<ui-hide-scrollbar,hide-scrollbar>>|Hide the main scrollbar.
|<<ui-css-media-type,css-media-type>>|Set the CSS media type. |<<ui-css-media-type,css-media-type>>|Set the CSS media type.
|<<ui-smooth-scrolling,smooth-scrolling>>|Whether to enable smooth scrolling for webpages. |<<ui-smooth-scrolling,smooth-scrolling>>|Whether to enable smooth scrolling for webpages.
|<<ui-remove-finished-downloads,remove-finished-downloads>>|Number of milliseconds to wait before removing finished downloads. Will not be removed if value is -1. |<<ui-remove-finished-downloads,remove-finished-downloads>>|Number of milliseconds to wait before removing finished downloads. Will not be removed if value is -1.
@ -618,11 +619,20 @@ This setting is only available with the QtWebKit backend.
[[ui-user-stylesheet]] [[ui-user-stylesheet]]
=== user-stylesheet === user-stylesheet
User stylesheet to use (absolute filename, filename relative to the config directory or CSS string). Will expand environment variables. User stylesheet to use (absolute filename or filename relative to the config directory). Will expand environment variables.
Default: +pass:[html &gt; ::-webkit-scrollbar { width: 0px; height: 0px; }]+ Default: empty
This setting is only available with the QtWebKit backend. [[ui-hide-scrollbar]]
=== hide-scrollbar
Hide the main scrollbar.
Valid values:
* +true+
* +false+
Default: +pass:[true]+
[[ui-css-media-type]] [[ui-css-media-type]]
=== css-media-type === css-media-type
@ -677,6 +687,7 @@ The format to use for the window title. The following placeholders are defined:
* `{id}`: The internal window ID of this window. * `{id}`: The internal window ID of this window.
* `{scroll_pos}`: The page scroll position. * `{scroll_pos}`: The page scroll position.
* `{host}`: The host of the current web page. * `{host}`: The host of the current web page.
* `{backend}`: Either 'webkit' or 'webengine'
Default: +pass:[{perc}{title}{title_sep}qutebrowser]+ Default: +pass:[{perc}{title}{title_sep}qutebrowser]+
@ -756,8 +767,6 @@ User agent to send. Empty to send the default.
Default: empty Default: empty
This setting is only available with the QtWebKit backend.
[[network-proxy]] [[network-proxy]]
=== proxy === proxy
The proxy to use. The proxy to use.
@ -1195,6 +1204,7 @@ The format to use for the tab title. The following placeholders are defined:
* `{id}`: The internal tab ID of this tab. * `{id}`: The internal tab ID of this tab.
* `{scroll_pos}`: The page scroll position. * `{scroll_pos}`: The page scroll position.
* `{host}`: The host of the current web page. * `{host}`: The host of the current web page.
* `{backend}`: Either 'webkit' or 'webengine'
Default: +pass:[{index}: {title}]+ Default: +pass:[{index}: {title}]+

View File

@ -38,17 +38,8 @@ show it.
*-h*, *--help*:: *-h*, *--help*::
show this help message and exit show this help message and exit
*-c* 'CONFDIR', *--confdir* 'CONFDIR'::
Set config directory (empty for no config storage).
*--datadir* 'DATADIR'::
Set data directory (empty for no data storage).
*--cachedir* 'CACHEDIR'::
Set cache directory (empty for no cache storage).
*--basedir* 'BASEDIR':: *--basedir* 'BASEDIR'::
Base directory for all storage. Other --*dir arguments are ignored if this is given. Base directory for all storage.
*-V*, *--version*:: *-V*, *--version*::
Show version and quit. Show version and quit.
@ -111,8 +102,8 @@ show it.
*--no-err-windows*:: *--no-err-windows*::
Don't show any error windows (used for tests/smoke.py). Don't show any error windows (used for tests/smoke.py).
*--qt-arg* 'QT_ARG':: *--qt-arg* 'NAME' 'VALUE'::
Pass an argument with a value to Qt. Pass an argument with a value to Qt. For example, you can do `--qt-arg geometry 650x555+200+300` to set the window geometry.
*--qt-flag* 'QT_FLAG':: *--qt-flag* 'QT_FLAG'::
Pass an argument to Qt as flag. Pass an argument to Qt as flag.

View File

@ -2,4 +2,4 @@
codecov==2.0.5 codecov==2.0.5
coverage==4.2 coverage==4.2
requests==2.11.1 requests==2.12.1

View File

@ -1,3 +1,3 @@
# This file is automatically generated by scripts/dev/recompile_requirements.py # This file is automatically generated by scripts/dev/recompile_requirements.py
cx-Freeze==4.3.4 cx-Freeze==4.3.4 # rq.filter: < 5.0.0

View File

@ -1 +1,5 @@
cx_Freeze cx-Freeze < 5.0.0
# We'll probably switch to PyInstaller soon, and 5.x doesn't install without a
# compiler?
#@ filter: cx-Freeze < 5.0.0

View File

@ -15,7 +15,7 @@ flake8-tuple==0.2.12
mccabe==0.5.2 mccabe==0.5.2
packaging==16.8 packaging==16.8
pep8-naming==0.4.1 pep8-naming==0.4.1
pycodestyle==2.1.0 pycodestyle==2.2.0
pydocstyle==1.1.1 pydocstyle==1.1.1
pyflakes==1.3.0 pyflakes==1.3.0
pyparsing==2.1.10 pyparsing==2.1.10

View File

@ -15,7 +15,7 @@ pydocstyle
pyflakes pyflakes
# Pinned to 2.0.0 otherwise # Pinned to 2.0.0 otherwise
pycodestyle==2.1.0 pycodestyle==2.2.0
# Waiting until flake8-putty updated # Waiting until flake8-putty updated
#@ filter: flake8 < 3.0.0 #@ filter: flake8 < 3.0.0

View File

@ -7,6 +7,6 @@ lazy-object-proxy==1.2.2
mccabe==0.5.2 mccabe==0.5.2
-e git+https://github.com/PyCQA/pylint.git#egg=pylint -e git+https://github.com/PyCQA/pylint.git#egg=pylint
./scripts/dev/pylint_checkers ./scripts/dev/pylint_checkers
requests==2.11.1 requests==2.12.1
six==1.10.0 six==1.10.0
wrapt==1.10.8 wrapt==1.10.8

View File

@ -7,7 +7,7 @@ lazy-object-proxy==1.2.2
mccabe==0.5.2 mccabe==0.5.2
pylint==1.6.4 pylint==1.6.4
./scripts/dev/pylint_checkers ./scripts/dev/pylint_checkers
requests==2.11.1 requests==2.12.1
six==1.10.0 six==1.10.0
uritemplate==3.0.0 uritemplate==3.0.0
uritemplate.py==3.0.2 uritemplate.py==3.0.2

View File

@ -20,7 +20,7 @@ pytest==3.0.4
pytest-bdd==2.18.1 pytest-bdd==2.18.1
pytest-catchlog==1.2.2 pytest-catchlog==1.2.2
pytest-cov==2.4.0 pytest-cov==2.4.0
pytest-faulthandler==1.3.0 pytest-faulthandler==1.3.1
pytest-instafail==0.3.0 pytest-instafail==0.3.0
pytest-mock==1.4.0 pytest-mock==1.4.0
pytest-qt==2.1.0 pytest-qt==2.1.0

View File

@ -2,5 +2,5 @@
pluggy==0.4.0 pluggy==0.4.0
py==1.4.31 py==1.4.31
tox==2.4.1 tox==2.5.0
virtualenv==15.0.3 virtualenv==15.1.0

View File

@ -17,7 +17,6 @@ markers =
qtwebengine_todo: Features still missing with QtWebEngine qtwebengine_todo: Features still missing with QtWebEngine
qtwebengine_skip: Tests not applicable with QtWebEngine qtwebengine_skip: Tests not applicable with QtWebEngine
qtwebkit_skip: Tests not applicable with QtWebKit qtwebkit_skip: Tests not applicable with QtWebKit
qtwebengine_createWindow: Tests using createWindow with QtWebEngine (QTBUG-54419)
qtwebengine_flaky: Tests which are flaky (and currently skipped) with QtWebEngine qtwebengine_flaky: Tests which are flaky (and currently skipped) with QtWebEngine
qtwebengine_osx_xfail: Tests which fail on OS X with QtWebEngine qtwebengine_osx_xfail: Tests which fail on OS X with QtWebEngine
js_prompt: Tests needing to display a javascript prompt js_prompt: Tests needing to display a javascript prompt

View File

@ -320,8 +320,8 @@ def _open_quickstart(args):
Args: Args:
args: The argparse namespace. args: The argparse namespace.
""" """
if args.datadir is not None or args.basedir is not None: if args.basedir is not None:
# With --datadir or --basedir given, don't open quickstart. # With --basedir given, don't open quickstart.
return return
state_config = objreg.get('state-config') state_config = objreg.get('state-config')
try: try:

View File

@ -29,7 +29,7 @@ import fnmatch
from qutebrowser.browser import downloads from qutebrowser.browser import downloads
from qutebrowser.config import config from qutebrowser.config import config
from qutebrowser.utils import objreg, standarddir, log, message from qutebrowser.utils import objreg, standarddir, log, message
from qutebrowser.commands import cmdutils, cmdexc from qutebrowser.commands import cmdutils
def guess_zip_filename(zf): def guess_zip_filename(zf):
@ -113,16 +113,10 @@ class HostBlocker:
self._done_count = 0 self._done_count = 0
data_dir = standarddir.data() data_dir = standarddir.data()
if data_dir is None:
self._local_hosts_file = None
else:
self._local_hosts_file = os.path.join(data_dir, 'blocked-hosts') self._local_hosts_file = os.path.join(data_dir, 'blocked-hosts')
self.on_config_changed() self.on_config_changed()
config_dir = standarddir.config() config_dir = standarddir.config()
if config_dir is None:
self._config_hosts_file = None
else:
self._config_hosts_file = os.path.join(config_dir, 'blocked-hosts') self._config_hosts_file = os.path.join(config_dir, 'blocked-hosts')
objreg.get('config').changed.connect(self.on_config_changed) objreg.get('config').changed.connect(self.on_config_changed)
@ -146,7 +140,7 @@ class HostBlocker:
Return: Return:
True if a read was attempted, False otherwise True if a read was attempted, False otherwise
""" """
if filename is None or not os.path.exists(filename): if not os.path.exists(filename):
return False return False
try: try:
@ -162,9 +156,6 @@ class HostBlocker:
"""Read hosts from the existing blocked-hosts file.""" """Read hosts from the existing blocked-hosts file."""
self._blocked_hosts = set() self._blocked_hosts = set()
if self._local_hosts_file is None:
return
self._read_hosts_file(self._config_hosts_file, self._read_hosts_file(self._config_hosts_file,
self._config_blocked_hosts) self._config_blocked_hosts)
@ -186,8 +177,6 @@ class HostBlocker:
""" """
self._read_hosts_file(self._config_hosts_file, self._read_hosts_file(self._config_hosts_file,
self._config_blocked_hosts) self._config_blocked_hosts)
if self._local_hosts_file is None:
raise cmdexc.CommandError("No data storage is configured!")
self._blocked_hosts = set() self._blocked_hosts = set()
self._done_count = 0 self._done_count = 0
urls = config.get('content', 'host-block-lists') urls = config.get('content', 'host-block-lists')
@ -275,7 +264,7 @@ class HostBlocker:
def on_config_changed(self): def on_config_changed(self):
"""Update files when the config changed.""" """Update files when the config changed."""
urls = config.get('content', 'host-block-lists') urls = config.get('content', 'host-block-lists')
if urls is None and self._local_hosts_file is not None: if urls is None:
try: try:
os.remove(self._local_hosts_file) os.remove(self._local_hosts_file)
except FileNotFoundError: except FileNotFoundError:

View File

@ -84,6 +84,7 @@ class TabData:
inspector: The QWebInspector used for this webview. inspector: The QWebInspector used for this webview.
viewing_source: Set if we're currently showing a source view. viewing_source: Set if we're currently showing a source view.
override_target: Override for open_target for fake clicks (like hints). override_target: Override for open_target for fake clicks (like hints).
Only used for QtWebKit.
pinned: Flag to pin the tab pinned: Flag to pin the tab
""" """

View File

@ -1539,7 +1539,7 @@ class CommandDispatcher:
message.error("Focused element is not editable!") message.error("Focused element is not editable!")
return return
text = elem.text(use_js=True) text = elem.value()
ed = editor.ExternalEditor(self._tabbed_browser) ed = editor.ExternalEditor(self._tabbed_browser)
ed.editing_finished.connect(functools.partial( ed.editing_finished.connect(functools.partial(
self.on_editing_finished, elem)) self.on_editing_finished, elem))
@ -1566,7 +1566,7 @@ class CommandDispatcher:
text: The new text to insert. text: The new text to insert.
""" """
try: try:
elem.set_text(text, use_js=True) elem.set_value(text)
except webelem.Error as e: except webelem.Error as e:
raise cmdexc.CommandError(str(e)) raise cmdexc.CommandError(str(e))
@ -1625,7 +1625,7 @@ class CommandDispatcher:
def single_cb(elem): def single_cb(elem):
"""Click a single element.""" """Click a single element."""
if elem is None: if elem is None:
message.error("No element found!") message.error("No element found with id {}!".format(value))
return return
try: try:
elem.click(target) elem.click(target)

View File

@ -88,11 +88,7 @@ class Entry:
if not url.isValid(): if not url.isValid():
raise ValueError("Invalid URL: {}".format(url.errorString())) raise ValueError("Invalid URL: {}".format(url.errorString()))
if atime.startswith('\0'): # https://github.com/The-Compiler/qutebrowser/issues/670
log.init.debug(
"Removing NUL bytes from entry {!r} - see "
"https://github.com/The-Compiler/qutebrowser/issues/"
"670".format(data))
atime = atime.lstrip('\0') atime = atime.lstrip('\0')
if '-' in atime: if '-' in atime:
@ -129,7 +125,6 @@ class WebHistory(QObject):
Attributes: Attributes:
history_dict: An OrderedDict of URLs read from the on-disk history. history_dict: An OrderedDict of URLs read from the on-disk history.
_hist_dir: The directory to store the history in
_lineparser: The AppendLineParser used to save the history. _lineparser: The AppendLineParser used to save the history.
_new_history: A list of Entry items of the current session. _new_history: A list of Entry items of the current session.
_saved_count: How many HistoryEntries have been written to disk. _saved_count: How many HistoryEntries have been written to disk.
@ -157,7 +152,6 @@ class WebHistory(QObject):
super().__init__(parent) super().__init__(parent)
self._initial_read_started = False self._initial_read_started = False
self._initial_read_done = False self._initial_read_done = False
self._hist_dir = hist_dir
self._lineparser = lineparser.AppendLineParser(hist_dir, hist_name, self._lineparser = lineparser.AppendLineParser(hist_dir, hist_name,
parent=self) parent=self)
self.history_dict = collections.OrderedDict() self.history_dict = collections.OrderedDict()
@ -183,12 +177,6 @@ class WebHistory(QObject):
return return
self._initial_read_started = True self._initial_read_started = True
if self._hist_dir is None:
self._initial_read_done = True
self.async_read_done.emit()
assert not self._temp_history
return
with self._lineparser.open(): with self._lineparser.open():
for line in self._lineparser: for line in self._lineparser:
yield yield

View File

@ -146,9 +146,13 @@ def ignore_certificate_errors(url, errors, abort_on):
""".strip()) """.strip())
msg = err_template.render(url=url, errors=errors) msg = err_template.render(url=url, errors=errors)
return message.ask(title="Certificate errors - continue?", text=msg, ignore = message.ask(title="Certificate errors - continue?", text=msg,
mode=usertypes.PromptMode.yesno, default=False, mode=usertypes.PromptMode.yesno, default=False,
abort_on=abort_on) abort_on=abort_on)
if ignore is None:
# prompt aborted
ignore = False
return ignore
elif ssl_strict is False: elif ssl_strict is False:
log.webview.debug("ssl-strict is False, only warning about errors") log.webview.debug("ssl-strict is False, only warning about errors")
for err in errors: for err in errors:
@ -222,3 +226,19 @@ def get_tab(win_id, target):
tabbed_browser = objreg.get('tabbed-browser', scope='window', tabbed_browser = objreg.get('tabbed-browser', scope='window',
window=win_id) window=win_id)
return tabbed_browser.tabopen(url=None, background=bg_tab) return tabbed_browser.tabopen(url=None, background=bg_tab)
def get_user_stylesheet():
"""Get the combined user-stylesheet."""
filename = config.get('ui', 'user-stylesheet')
if filename is None:
css = ''
else:
with open(filename, 'r', encoding='utf-8') as f:
css = f.read()
if config.get('ui', 'hide-scrollbar'):
css += '\nhtml > ::-webkit-scrollbar { width: 0px; height: 0px; }'
return css

View File

@ -73,8 +73,7 @@ class UrlMarkManager(QObject):
Attributes: Attributes:
marks: An OrderedDict of all quickmarks/bookmarks. marks: An OrderedDict of all quickmarks/bookmarks.
_lineparser: The LineParser used for the marks, or None _lineparser: The LineParser used for the marks
(when qutebrowser is started with -c '').
Signals: Signals:
changed: Emitted when anything changed. changed: Emitted when anything changed.
@ -91,10 +90,6 @@ class UrlMarkManager(QObject):
super().__init__(parent) super().__init__(parent)
self.marks = collections.OrderedDict() self.marks = collections.OrderedDict()
self._lineparser = None
if standarddir.config() is None:
return
self._init_lineparser() self._init_lineparser()
for line in self._lineparser: for line in self._lineparser:
@ -115,9 +110,7 @@ class UrlMarkManager(QObject):
def save(self): def save(self):
"""Save the marks to disk.""" """Save the marks to disk."""
if self._lineparser is not None: self._lineparser.data = [' '.join(tpl) for tpl in self.marks.items()]
self._lineparser.data = [' '.join(tpl)
for tpl in self.marks.items()]
self._lineparser.save() self._lineparser.save()
def delete(self, key): def delete(self, key):

View File

@ -87,7 +87,7 @@ class AbstractWebElement(collections.abc.MutableMapping):
raise NotImplementedError raise NotImplementedError
def __str__(self): def __str__(self):
return self.text() raise NotImplementedError
def __getitem__(self, key): def __getitem__(self, key):
raise NotImplementedError raise NotImplementedError
@ -138,24 +138,12 @@ class AbstractWebElement(collections.abc.MutableMapping):
"""Get the full HTML representation of this element.""" """Get the full HTML representation of this element."""
raise NotImplementedError raise NotImplementedError
def text(self, *, use_js=False): def value(self):
"""Get the plain text content for this element. """Get the value attribute for this element."""
Args:
use_js: Whether to use javascript if the element isn't
content-editable.
"""
# FIXME:qtwebengine what to do about use_js with WebEngine?
raise NotImplementedError raise NotImplementedError
def set_text(self, text, *, use_js=False): def set_value(self, value):
"""Set the given plain text. """Set the element value."""
Args:
use_js: Whether to use javascript if the element isn't
content-editable.
"""
# FIXME:qtwebengine what to do about use_js with WebEngine?
raise NotImplementedError raise NotImplementedError
def insert_text(self, text): def insert_text(self, text):
@ -338,6 +326,8 @@ class AbstractWebElement(collections.abc.MutableMapping):
"""Simulate a click on the element.""" """Simulate a click on the element."""
# FIXME:qtwebengine do we need this? # FIXME:qtwebengine do we need this?
# self._widget.setFocus() # self._widget.setFocus()
# For QtWebKit
self._tab.data.override_target = click_target self._tab.data.override_target = click_target
pos = self._mouse_pos() pos = self._mouse_pos()
@ -345,20 +335,24 @@ class AbstractWebElement(collections.abc.MutableMapping):
log.webelem.debug("Sending fake click to {!r} at position {} with " log.webelem.debug("Sending fake click to {!r} at position {} with "
"target {}".format(self, pos, click_target)) "target {}".format(self, pos, click_target))
if click_target in [usertypes.ClickTarget.tab, modifiers = {
usertypes.ClickTarget.tab_bg, usertypes.ClickTarget.normal: Qt.NoModifier,
usertypes.ClickTarget.window]: usertypes.ClickTarget.window: Qt.AltModifier | Qt.ShiftModifier,
modifiers = Qt.ControlModifier usertypes.ClickTarget.tab: Qt.ControlModifier,
usertypes.ClickTarget.tab_bg: Qt.ControlModifier,
}
if config.get('tabs', 'background-tabs'):
modifiers[usertypes.ClickTarget.tab] |= Qt.ShiftModifier
else: else:
modifiers = Qt.NoModifier modifiers[usertypes.ClickTarget.tab_bg] |= Qt.ShiftModifier
events = [ events = [
QMouseEvent(QEvent.MouseMove, pos, Qt.NoButton, Qt.NoButton, QMouseEvent(QEvent.MouseMove, pos, Qt.NoButton, Qt.NoButton,
Qt.NoModifier), Qt.NoModifier),
QMouseEvent(QEvent.MouseButtonPress, pos, Qt.LeftButton, QMouseEvent(QEvent.MouseButtonPress, pos, Qt.LeftButton,
Qt.LeftButton, modifiers), Qt.LeftButton, modifiers[click_target]),
QMouseEvent(QEvent.MouseButtonRelease, pos, Qt.LeftButton, QMouseEvent(QEvent.MouseButtonRelease, pos, Qt.LeftButton,
Qt.NoButton, modifiers), Qt.NoButton, modifiers[click_target]),
] ]
for evt in events: for evt in events:

View File

@ -23,6 +23,7 @@
from PyQt5.QtWebEngineCore import QWebEngineUrlRequestInterceptor from PyQt5.QtWebEngineCore import QWebEngineUrlRequestInterceptor
# pylint: enable=no-name-in-module,import-error,useless-suppression # pylint: enable=no-name-in-module,import-error,useless-suppression
from qutebrowser.config import config
from qutebrowser.browser import shared from qutebrowser.browser import shared
from qutebrowser.utils import utils, log from qutebrowser.utils import utils, log
@ -64,3 +65,7 @@ class RequestInterceptor(QWebEngineUrlRequestInterceptor):
for header, value in shared.custom_headers(): for header, value in shared.custom_headers():
info.setHttpHeader(header, value) info.setHttpHeader(header, value)
user_agent = config.get('network', 'user-agent')
if user_agent is not None:
info.setHttpHeader(b'User-Agent', user_agent.encode('ascii'))

View File

@ -37,6 +37,9 @@ class WebEngineElement(webelem.AbstractWebElement):
self._id = js_dict['id'] self._id = js_dict['id']
self._js_dict = js_dict self._js_dict = js_dict
def __str__(self):
return self._js_dict.get('text', '')
def __eq__(self, other): def __eq__(self, other):
if not isinstance(other, WebEngineElement): if not isinstance(other, WebEngineElement):
return NotImplemented return NotImplemented
@ -87,27 +90,11 @@ class WebEngineElement(webelem.AbstractWebElement):
"""Get the full HTML representation of this element.""" """Get the full HTML representation of this element."""
return self._js_dict['outer_xml'] return self._js_dict['outer_xml']
def text(self, *, use_js=False): def value(self):
"""Get the plain text content for this element. return self._js_dict['value']
Args: def set_value(self, value):
use_js: Whether to use javascript if the element isn't js_code = javascript.assemble('webelem', 'set_value', self._id, value)
content-editable.
"""
if use_js:
# FIXME:qtwebengine what to do about use_js with WebEngine?
log.stub('with use_js=True')
return self._js_dict.get('text', '')
def set_text(self, text, *, use_js=False):
"""Set the given plain text.
Args:
use_js: Whether to use javascript if the element isn't
content-editable.
"""
# FIXME:qtwebengine what to do about use_js with WebEngine?
js_code = javascript.assemble('webelem', 'set_text', self._id, text)
self._tab.run_js_async(js_code) self._tab.run_js_async(js_code)
def insert_text(self, text): def insert_text(self, text):

View File

@ -27,11 +27,13 @@ Module attributes:
import os import os
# pylint: disable=no-name-in-module,import-error,useless-suppression # pylint: disable=no-name-in-module,import-error,useless-suppression
from PyQt5.QtWebEngineWidgets import QWebEngineSettings from PyQt5.QtWebEngineWidgets import (QWebEngineSettings, QWebEngineProfile,
QWebEngineScript)
# pylint: enable=no-name-in-module,import-error,useless-suppression # pylint: enable=no-name-in-module,import-error,useless-suppression
from qutebrowser.browser import shared
from qutebrowser.config import websettings, config from qutebrowser.config import websettings, config
from qutebrowser.utils import objreg, utils from qutebrowser.utils import objreg, utils, standarddir, javascript
class Attribute(websettings.Attribute): class Attribute(websettings.Attribute):
@ -63,19 +65,59 @@ class StaticSetter(websettings.StaticSetter):
GLOBAL_SETTINGS = QWebEngineSettings.globalSettings GLOBAL_SETTINGS = QWebEngineSettings.globalSettings
def _init_stylesheet(profile):
"""Initialize custom stylesheets.
Mostly inspired by QupZilla:
https://github.com/QupZilla/qupzilla/blob/v2.0/src/lib/app/mainapplication.cpp#L1063-L1101
https://github.com/QupZilla/qupzilla/blob/v2.0/src/lib/tools/scripts.cpp#L119-L132
FIXME:qtwebengine Use QWebEngineStyleSheet once that's available
https://codereview.qt-project.org/#/c/148671/
"""
old_script = profile.scripts().findScript('_qute_stylesheet')
if not old_script.isNull():
profile.scripts().remove(old_script)
css = shared.get_user_stylesheet()
source = """
(function() {{
var css = document.createElement('style');
css.setAttribute('type', 'text/css');
css.appendChild(document.createTextNode('{}'));
document.getElementsByTagName('head')[0].appendChild(css);
}})()
""".format(javascript.string_escape(css))
script = QWebEngineScript()
script.setName('_qute_stylesheet')
script.setInjectionPoint(QWebEngineScript.DocumentReady)
script.setWorldId(QWebEngineScript.ApplicationWorld)
script.setRunsOnSubFrames(True)
script.setSourceCode(source)
profile.scripts().insert(script)
def update_settings(section, option): def update_settings(section, option):
"""Update global settings when qwebsettings changed.""" """Update global settings when qwebsettings changed."""
websettings.update_mappings(MAPPINGS, section, option) websettings.update_mappings(MAPPINGS, section, option)
profile = QWebEngineProfile.defaultProfile()
if section == 'ui' and option in ['hide-scrollbar', 'user-stylesheet']:
_init_stylesheet(profile)
def init(): def init():
"""Initialize the global QWebSettings.""" """Initialize the global QWebSettings."""
# FIXME:qtwebengine set paths in profile
if config.get('general', 'developer-extras'): if config.get('general', 'developer-extras'):
# FIXME:qtwebengine Make sure we call globalSettings *after* this... # FIXME:qtwebengine Make sure we call globalSettings *after* this...
os.environ['QTWEBENGINE_REMOTE_DEBUGGING'] = str(utils.random_port()) os.environ['QTWEBENGINE_REMOTE_DEBUGGING'] = str(utils.random_port())
profile = QWebEngineProfile.defaultProfile()
profile.setCachePath(os.path.join(standarddir.cache(), 'webengine'))
profile.setPersistentStoragePath(
os.path.join(standarddir.data(), 'webengine'))
_init_stylesheet(profile)
websettings.init_mappings(MAPPINGS) websettings.init_mappings(MAPPINGS)
objreg.get('config').changed.connect(update_settings) objreg.get('config').changed.connect(update_settings)
@ -98,18 +140,12 @@ def shutdown():
# - PictographFont # - PictographFont
# #
# TODO settings on profile: # TODO settings on profile:
# - cachePath
# - httpAcceptLanguage
# - httpCacheMaximumSize # - httpCacheMaximumSize
# - httpUserAgent
# - persistentCookiesPolicy # - persistentCookiesPolicy
# - offTheRecord # - offTheRecord
# - persistentStoragePath
# #
# TODO settings elsewhere: # TODO settings elsewhere:
# - proxy # - proxy
# - custom headers
# - ssl-strict
MAPPINGS = { MAPPINGS = {
'content': { 'content': {

View File

@ -469,7 +469,6 @@ class WebEngineTab(browsertab.AbstractTab):
self._set_widget(widget) self._set_widget(widget)
self._connect_signals() self._connect_signals()
self.backend = usertypes.Backend.QtWebEngine self.backend = usertypes.Backend.QtWebEngine
# init js stuff
self._init_js() self._init_js()
self._child_event_filter = None self._child_event_filter = None
self.needs_qtbug54419_workaround = False self.needs_qtbug54419_workaround = False

View File

@ -74,18 +74,19 @@ class WebEngineView(QWebEngineView):
""" """
debug_type = debug.qenum_key(QWebEnginePage, wintype) debug_type = debug.qenum_key(QWebEnginePage, wintype)
background_tabs = config.get('tabs', 'background-tabs') background_tabs = config.get('tabs', 'background-tabs')
override_target = self._tabdata.override_target
log.webview.debug("createWindow with type {}, background_tabs " log.webview.debug("createWindow with type {}, background_tabs "
"{}, override_target {}".format( "{}".format(debug_type, background_tabs))
debug_type, background_tabs, override_target))
if override_target is not None: try:
target = override_target background_tab_wintype = QWebEnginePage.WebBrowserBackgroundTab
self._tabdata.override_target = None except AttributeError:
elif wintype == QWebEnginePage.WebBrowserWindow: # This is unavailable with an older PyQt, but we still might get
log.webview.debug("createWindow with WebBrowserWindow - when does " # this with a newer Qt...
"this happen?!") background_tab_wintype = 0x0003
if wintype == QWebEnginePage.WebBrowserWindow:
# Shift-Alt-Click
target = usertypes.ClickTarget.window target = usertypes.ClickTarget.window
elif wintype == QWebEnginePage.WebDialog: elif wintype == QWebEnginePage.WebDialog:
log.webview.warning("{} requested, but we don't support " log.webview.warning("{} requested, but we don't support "
@ -93,12 +94,12 @@ class WebEngineView(QWebEngineView):
target = usertypes.ClickTarget.tab target = usertypes.ClickTarget.tab
elif wintype == QWebEnginePage.WebBrowserTab: elif wintype == QWebEnginePage.WebBrowserTab:
# Middle-click / Ctrl-Click with Shift # Middle-click / Ctrl-Click with Shift
# FIXME:qtwebengine this also affects target=_blank links...
if background_tabs: if background_tabs:
target = usertypes.ClickTarget.tab target = usertypes.ClickTarget.tab
else: else:
target = usertypes.ClickTarget.tab_bg target = usertypes.ClickTarget.tab_bg
elif (hasattr(QWebEnginePage, 'WebBrowserBackgroundTab') and elif wintype == background_tab_wintype:
wintype == QWebEnginePage.WebBrowserBackgroundTab):
# Middle-click / Ctrl-Click # Middle-click / Ctrl-Click
if background_tabs: if background_tabs:
target = usertypes.ClickTarget.tab_bg target = usertypes.ClickTarget.tab_bg

View File

@ -32,23 +32,14 @@ class DiskCache(QNetworkDiskCache):
"""Disk cache which sets correct cache dir and size. """Disk cache which sets correct cache dir and size.
If the cache is deactivated via the command line argument --cachedir="",
both attributes _cache_dir and _http_cache_dir are set to None.
Attributes: Attributes:
_activated: Whether the cache should be used. _activated: Whether the cache should be used.
_cache_dir: The base directory for cache files (standarddir.cache()) or _cache_dir: The base directory for cache files (standarddir.cache())
None.
_http_cache_dir: the HTTP subfolder in _cache_dir or None.
""" """
def __init__(self, cache_dir, parent=None): def __init__(self, cache_dir, parent=None):
super().__init__(parent) super().__init__(parent)
self._cache_dir = cache_dir self._cache_dir = cache_dir
if cache_dir is None:
self._http_cache_dir = None
else:
self._http_cache_dir = os.path.join(cache_dir, 'http')
self._maybe_activate() self._maybe_activate()
objreg.get('config').changed.connect(self.on_config_changed) objreg.get('config').changed.connect(self.on_config_changed)
@ -59,12 +50,11 @@ class DiskCache(QNetworkDiskCache):
def _maybe_activate(self): def _maybe_activate(self):
"""Activate/deactivate the cache based on the config.""" """Activate/deactivate the cache based on the config."""
if (config.get('general', 'private-browsing') or if config.get('general', 'private-browsing'):
self._cache_dir is None):
self._activated = False self._activated = False
else: else:
self._activated = True self._activated = True
self.setCacheDirectory(self._http_cache_dir) self.setCacheDirectory(os.path.join(self._cache_dir, 'http'))
self.setMaximumCacheSize(config.get('storage', 'cache-size')) self.setMaximumCacheSize(config.get('storage', 'cache-size'))
@pyqtSlot(str, str) @pyqtSlot(str, str)

View File

@ -46,6 +46,10 @@ class WebKitElement(webelem.AbstractWebElement):
raise IsNullError('{} is a null element!'.format(elem)) raise IsNullError('{} is a null element!'.format(elem))
self._elem = elem self._elem = elem
def __str__(self):
self._check_vanished()
return self._elem.toPlainText()
def __eq__(self, other): def __eq__(self, other):
if not isinstance(other, WebKitElement): if not isinstance(other, WebKitElement):
return NotImplemented return NotImplemented
@ -116,22 +120,19 @@ class WebKitElement(webelem.AbstractWebElement):
self._check_vanished() self._check_vanished()
return self._elem.toOuterXml() return self._elem.toOuterXml()
def text(self, *, use_js=False): def value(self):
self._check_vanished() self._check_vanished()
if self.is_content_editable() or not use_js:
return self._elem.toPlainText()
else:
return self._elem.evaluateJavaScript('this.value') return self._elem.evaluateJavaScript('this.value')
def set_text(self, text, *, use_js=False): def set_value(self, value):
self._check_vanished() self._check_vanished()
if self.is_content_editable() or not use_js: if self.is_content_editable():
log.webelem.debug("Filling {!r} via set_text.".format(self)) log.webelem.debug("Filling {!r} via set_text.".format(self))
self._elem.setPlainText(text) self._elem.setPlainText(value)
else: else:
log.webelem.debug("Filling {!r} via javascript.".format(self)) log.webelem.debug("Filling {!r} via javascript.".format(self))
text = javascript.string_escape(text) value = javascript.string_escape(value)
self._elem.evaluateJavaScript("this.value='{}'".format(text)) self._elem.evaluateJavaScript("this.value='{}'".format(value))
def insert_text(self, text): def insert_text(self, text):
self._check_vanished() self._check_vanished()

View File

@ -29,7 +29,8 @@ import os.path
from PyQt5.QtWebKit import QWebSettings from PyQt5.QtWebKit import QWebSettings
from qutebrowser.config import config, websettings from qutebrowser.config import config, websettings
from qutebrowser.utils import standarddir, objreg from qutebrowser.utils import standarddir, objreg, urlutils
from qutebrowser.browser import shared
class Attribute(websettings.Attribute): class Attribute(websettings.Attribute):
@ -80,14 +81,24 @@ class CookiePolicy(websettings.Base):
self.MAPPING[value]) self.MAPPING[value])
def _set_user_stylesheet():
"""Set the generated user-stylesheet."""
stylesheet = shared.get_user_stylesheet().encode('utf-8')
url = urlutils.data_url('text/css;charset=utf-8', stylesheet)
QWebSettings.globalSettings().setUserStyleSheetUrl(url)
def update_settings(section, option): def update_settings(section, option):
"""Update global settings when qwebsettings changed.""" """Update global settings when qwebsettings changed."""
cache_path = standarddir.cache()
if (section, option) == ('general', 'private-browsing'): if (section, option) == ('general', 'private-browsing'):
cache_path = standarddir.cache()
if config.get('general', 'private-browsing') or cache_path is None: if config.get('general', 'private-browsing') or cache_path is None:
QWebSettings.setIconDatabasePath('') QWebSettings.setIconDatabasePath('')
else: else:
QWebSettings.setIconDatabasePath(cache_path) QWebSettings.setIconDatabasePath(cache_path)
elif section == 'ui' and option in ['hide-scrollbar', 'user-stylesheet']:
_set_user_stylesheet()
websettings.update_mappings(MAPPINGS, section, option) websettings.update_mappings(MAPPINGS, section, option)
@ -95,20 +106,20 @@ def init():
"""Initialize the global QWebSettings.""" """Initialize the global QWebSettings."""
cache_path = standarddir.cache() cache_path = standarddir.cache()
data_path = standarddir.data() data_path = standarddir.data()
if config.get('general', 'private-browsing') or cache_path is None: if config.get('general', 'private-browsing'):
QWebSettings.setIconDatabasePath('') QWebSettings.setIconDatabasePath('')
else: else:
QWebSettings.setIconDatabasePath(cache_path) QWebSettings.setIconDatabasePath(cache_path)
if cache_path is not None:
QWebSettings.setOfflineWebApplicationCachePath( QWebSettings.setOfflineWebApplicationCachePath(
os.path.join(cache_path, 'application-cache')) os.path.join(cache_path, 'application-cache'))
if data_path is not None:
QWebSettings.globalSettings().setLocalStoragePath( QWebSettings.globalSettings().setLocalStoragePath(
os.path.join(data_path, 'local-storage')) os.path.join(data_path, 'local-storage'))
QWebSettings.setOfflineStoragePath( QWebSettings.setOfflineStoragePath(
os.path.join(data_path, 'offline-storage')) os.path.join(data_path, 'offline-storage'))
websettings.init_mappings(MAPPINGS) websettings.init_mappings(MAPPINGS)
_set_user_stylesheet()
objreg.get('config').changed.connect(update_settings) objreg.get('config').changed.connect(update_settings)
@ -205,9 +216,7 @@ MAPPINGS = {
Attribute(QWebSettings.ZoomTextOnly), Attribute(QWebSettings.ZoomTextOnly),
'frame-flattening': 'frame-flattening':
Attribute(QWebSettings.FrameFlatteningEnabled), Attribute(QWebSettings.FrameFlatteningEnabled),
'user-stylesheet': # user-stylesheet is handled separately
Setter(getter=QWebSettings.userStyleSheetUrl,
setter=QWebSettings.setUserStyleSheetUrl),
'css-media-type': 'css-media-type':
NullStringSetter(getter=QWebSettings.cssMediaType, NullStringSetter(getter=QWebSettings.cssMediaType,
setter=QWebSettings.setCSSMediaType), setter=QWebSettings.setCSSMediaType),

View File

@ -408,15 +408,11 @@ def run_async(tab, cmd, *args, win_id, env, verbose=False):
user_agent = config.get('network', 'user-agent') user_agent = config.get('network', 'user-agent')
if user_agent is not None: if user_agent is not None:
env['QUTE_USER_AGENT'] = user_agent env['QUTE_USER_AGENT'] = user_agent
config_dir = standarddir.config()
if config_dir is not None: env['QUTE_CONFIG_DIR'] = standarddir.config()
env['QUTE_CONFIG_DIR'] = config_dir env['QUTE_DATA_DIR'] = standarddir.data()
data_dir = standarddir.data() env['QUTE_DOWNLOAD_DIR'] = downloads.download_dir()
if data_dir is not None:
env['QUTE_DATA_DIR'] = data_dir
download_dir = downloads.download_dir()
if download_dir is not None:
env['QUTE_DOWNLOAD_DIR'] = download_dir
cmd_path = os.path.expanduser(cmd) cmd_path = os.path.expanduser(cmd)
# if cmd is not given as an absolute path, look it up # if cmd is not given as an absolute path, look it up

View File

@ -160,7 +160,6 @@ def _init_main_config(parent=None):
sys.exit(usertypes.Exit.err_config) sys.exit(usertypes.Exit.err_config)
else: else:
objreg.register('config', config_obj) objreg.register('config', config_obj)
if standarddir.config() is not None:
filename = os.path.join(standarddir.config(), 'qutebrowser.conf') filename = os.path.join(standarddir.config(), 'qutebrowser.conf')
save_manager = objreg.get('save-manager') save_manager = objreg.get('save-manager')
save_manager.add_saveable( save_manager.add_saveable(
@ -197,7 +196,6 @@ def _init_key_config(parent):
sys.exit(usertypes.Exit.err_key_config) sys.exit(usertypes.Exit.err_key_config)
else: else:
objreg.register('key-config', key_config) objreg.register('key-config', key_config)
if standarddir.config() is not None:
save_manager = objreg.get('save-manager') save_manager = objreg.get('save-manager')
filename = os.path.join(standarddir.config(), 'keys.conf') filename = os.path.join(standarddir.config(), 'keys.conf')
save_manager.add_saveable( save_manager.add_saveable(
@ -237,9 +235,6 @@ def _init_misc():
# This fixes one of the corruption issues here: # This fixes one of the corruption issues here:
# https://github.com/The-Compiler/qutebrowser/issues/515 # https://github.com/The-Compiler/qutebrowser/issues/515
if standarddir.config() is None:
path = os.devnull
else:
path = os.path.join(standarddir.config(), 'qsettings') path = os.path.join(standarddir.config(), 'qsettings')
for fmt in [QSettings.NativeFormat, QSettings.IniFormat]: for fmt in [QSettings.NativeFormat, QSettings.IniFormat]:
QSettings.setPath(fmt, QSettings.UserScope, path) QSettings.setPath(fmt, QSettings.UserScope, path)
@ -442,6 +437,11 @@ class ConfigManager(QObject):
('fonts', 'hints'): _transform_hint_font, ('fonts', 'hints'): _transform_hint_font,
('completion', 'show'): ('completion', 'show'):
_get_value_transformer({'false': 'never', 'true': 'always'}), _get_value_transformer({'false': 'never', 'true': 'always'}),
('ui', 'user-stylesheet'):
_get_value_transformer({
'html > ::-webkit-scrollbar { width: 0px; height: 0px; }': '',
'::-webkit-scrollbar { width: 0px; height: 0px; }': '',
}),
} }
changed = pyqtSignal(str, str) changed = pyqtSignal(str, str)
@ -659,10 +659,6 @@ class ConfigManager(QObject):
def read(self, configdir, fname, relaxed=False): def read(self, configdir, fname, relaxed=False):
"""Read the config from the given directory/file.""" """Read the config from the given directory/file."""
self._fname = fname self._fname = fname
if configdir is None:
self._configdir = None
self._initialized = True
else:
self._configdir = configdir self._configdir = configdir
parser = ini.ReadConfigParser(configdir, fname) parser = ini.ReadConfigParser(configdir, fname)
self._from_cp(parser, relaxed) self._from_cp(parser, relaxed)
@ -884,8 +880,6 @@ class ConfigManager(QObject):
def save(self): def save(self):
"""Save the config file.""" """Save the config file."""
if self._configdir is None:
return
configfile = os.path.join(self._configdir, self._fname) configfile = os.path.join(self._configdir, self._fname)
log.destroy.debug("Saving config to {}".format(configfile)) log.destroy.debug("Saving config to {}".format(configfile))
with qtutils.savefile_open(configfile) as f: with qtutils.savefile_open(configfile) as f:

View File

@ -323,13 +323,13 @@ def data(readonly=False):
"page."), "page."),
('user-stylesheet', ('user-stylesheet',
SettingValue(typ.UserStyleSheet(none_ok=True), SettingValue(typ.File(none_ok=True), ''),
'html > ::-webkit-scrollbar { width: 0px; ' "User stylesheet to use (absolute filename or filename relative "
'height: 0px; }', "to the config directory). Will expand environment variables."),
backends=[usertypes.Backend.QtWebKit]),
"User stylesheet to use (absolute filename, filename relative to " ('hide-scrollbar',
"the config directory or CSS string). Will expand environment " SettingValue(typ.Bool(), 'true'),
"variables."), "Hide the main scrollbar."),
('css-media-type', ('css-media-type',
SettingValue(typ.String(none_ok=True), '', SettingValue(typ.String(none_ok=True), '',
@ -356,7 +356,8 @@ def data(readonly=False):
('window-title-format', ('window-title-format',
SettingValue(typ.FormatString(fields=['perc', 'perc_raw', 'title', SettingValue(typ.FormatString(fields=['perc', 'perc_raw', 'title',
'title_sep', 'id', 'title_sep', 'id',
'scroll_pos', 'host']), 'scroll_pos', 'host',
'backend']),
'{perc}{title}{title_sep}qutebrowser'), '{perc}{title}{title_sep}qutebrowser'),
"The format to use for the window title. The following " "The format to use for the window title. The following "
"placeholders are defined:\n\n" "placeholders are defined:\n\n"
@ -367,7 +368,8 @@ def data(readonly=False):
"otherwise.\n" "otherwise.\n"
"* `{id}`: The internal window ID of this window.\n" "* `{id}`: The internal window ID of this window.\n"
"* `{scroll_pos}`: The page scroll position.\n" "* `{scroll_pos}`: The page scroll position.\n"
"* `{host}`: The host of the current web page."), "* `{host}`: The host of the current web page.\n"
"* `{backend}`: Either 'webkit' or 'webengine'"),
('modal-js-dialog', ('modal-js-dialog',
SettingValue(typ.Bool(), 'false'), SettingValue(typ.Bool(), 'false'),
@ -413,8 +415,7 @@ def data(readonly=False):
"Send the Referer header"), "Send the Referer header"),
('user-agent', ('user-agent',
SettingValue(typ.UserAgent(none_ok=True), '', SettingValue(typ.UserAgent(none_ok=True), ''),
backends=[usertypes.Backend.QtWebKit]),
"User agent to send. Empty to send the default."), "User agent to send. Empty to send the default."),
('proxy', ('proxy',
@ -681,7 +682,8 @@ def data(readonly=False):
"* `{index}`: The index of this tab.\n" "* `{index}`: The index of this tab.\n"
"* `{id}`: The internal tab ID of this tab.\n" "* `{id}`: The internal tab ID of this tab.\n"
"* `{scroll_pos}`: The page scroll position.\n" "* `{scroll_pos}`: The page scroll position.\n"
"* `{host}`: The host of the current web page."), "* `{host}`: The host of the current web page.\n"
"* `{backend}`: Either 'webkit' or 'webengine'"),
('title-format-pinned', ('title-format-pinned',
SettingValue(typ.FormatString( SettingValue(typ.FormatString(

View File

@ -22,7 +22,6 @@
import re import re
import json import json
import shlex import shlex
import base64
import codecs import codecs
import os.path import os.path
import itertools import itertools
@ -859,9 +858,7 @@ class File(BaseType):
value = os.path.expanduser(value) value = os.path.expanduser(value)
value = os.path.expandvars(value) value = os.path.expandvars(value)
if not os.path.isabs(value): if not os.path.isabs(value):
cfgdir = standarddir.config() value = os.path.join(standarddir.config(), value)
assert cfgdir is not None
value = os.path.join(cfgdir, value)
return value return value
def validate(self, value): def validate(self, value):
@ -872,12 +869,7 @@ class File(BaseType):
value = os.path.expandvars(value) value = os.path.expandvars(value)
try: try:
if not os.path.isabs(value): if not os.path.isabs(value):
cfgdir = standarddir.config() value = os.path.join(standarddir.config(), value)
if cfgdir is None:
raise configexc.ValidationError(
value, "must be an absolute path when not using a "
"config directory!")
value = os.path.join(cfgdir, value)
not_isfile_message = ("must be a valid path relative to the " not_isfile_message = ("must be a valid path relative to the "
"config directory!") "config directory!")
else: else:
@ -1172,48 +1164,6 @@ class Encoding(BaseType):
raise configexc.ValidationError(value, "is not a valid encoding!") raise configexc.ValidationError(value, "is not a valid encoding!")
class UserStyleSheet(File):
"""QWebSettings UserStyleSheet."""
def transform(self, value):
if not value:
return None
if standarddir.config() is None:
# We can't call super().transform() here as this counts on the
# validation previously ensuring that we don't have a relative path
# when starting with -c "".
path = None
else:
path = super().transform(value)
if path is not None and os.path.exists(path):
return QUrl.fromLocalFile(path)
else:
data = base64.b64encode(value.encode('utf-8')).decode('ascii')
return QUrl("data:text/css;charset=utf-8;base64,{}".format(data))
def validate(self, value):
self._basic_validation(value)
if not value:
return
value = os.path.expandvars(value)
value = os.path.expanduser(value)
try:
super().validate(value)
except configexc.ValidationError:
try:
if not os.path.isabs(value):
# probably a CSS, so we don't handle it as filename.
# FIXME We just try if it is encodable, maybe we should
# validate CSS?
# https://github.com/The-Compiler/qutebrowser/issues/115
value.encode('utf-8')
except UnicodeEncodeError as e:
raise configexc.ValidationError(value, str(e))
class AutoSearch(BaseType): class AutoSearch(BaseType):
"""Whether to start a search when something else than a URL is entered.""" """Whether to start a search when something else than a URL is entered."""
@ -1476,6 +1426,11 @@ class UserAgent(BaseType):
def validate(self, value): def validate(self, value):
self._basic_validation(value) self._basic_validation(value)
try:
value.encode('ascii')
except UnicodeEncodeError as e:
msg = "User-Agent contains non-ascii characters: {}".format(e)
raise configexc.ValidationError(value, msg)
# To update the following list of user agents, run the script 'ua_fetch.py' # To update the following list of user agents, run the script 'ua_fetch.py'
# Vim-protip: Place your cursor below this comment and run # Vim-protip: Place your cursor below this comment and run

View File

@ -47,14 +47,11 @@ class ReadConfigParser(configparser.ConfigParser):
self.optionxform = lambda opt: opt # be case-insensitive self.optionxform = lambda opt: opt # be case-insensitive
self._configdir = configdir self._configdir = configdir
self._fname = fname self._fname = fname
if self._configdir is None:
self._configfile = None
return
self._configfile = os.path.join(self._configdir, fname) self._configfile = os.path.join(self._configdir, fname)
if not os.path.isfile(self._configfile): if not os.path.isfile(self._configfile):
return return
log.init.debug("Reading config from {}".format(self._configfile)) log.init.debug("Reading config from {}".format(self._configfile))
if self._configfile is not None:
self.read(self._configfile, encoding='utf-8') self.read(self._configfile, encoding='utf-8')
def __repr__(self): def __repr__(self):

View File

@ -89,11 +89,9 @@ class KeyConfigParser(QObject):
self._cur_command = None self._cur_command = None
# Mapping of section name(s) to key binding -> command dicts. # Mapping of section name(s) to key binding -> command dicts.
self.keybindings = collections.OrderedDict() self.keybindings = collections.OrderedDict()
if configdir is None:
self._configfile = None
else:
self._configfile = os.path.join(configdir, fname) self._configfile = os.path.join(configdir, fname)
if self._configfile is None or not os.path.exists(self._configfile):
if not os.path.exists(self._configfile):
self._load_default() self._load_default()
else: else:
self._read(relaxed) self._read(relaxed)
@ -143,8 +141,6 @@ class KeyConfigParser(QObject):
def save(self): def save(self):
"""Save the key config file.""" """Save the key config file."""
if self._configfile is None:
return
log.destroy.debug("Saving key config to {}".format(self._configfile)) log.destroy.debug("Saving key config to {}".format(self._configfile))
with qtutils.savefile_open(self._configfile, encoding='utf-8') as f: with qtutils.savefile_open(self._configfile, encoding='utf-8') as f:
data = str(self) data = str(self)

View File

@ -34,6 +34,7 @@ window._qutebrowser.webelem = (function() {
var out = { var out = {
"id": id, "id": id,
"text": elem.text, "text": elem.text,
"value": elem.value,
"tag_name": elem.tagName, "tag_name": elem.tagName,
"outer_xml": elem.outerHTML, "outer_xml": elem.outerHTML,
"class_name": elem.className, "class_name": elem.className,
@ -129,8 +130,8 @@ window._qutebrowser.webelem = (function() {
return serialize_elem(elem); return serialize_elem(elem);
}; };
funcs.set_text = function(id, text) { funcs.set_value = function(id, value) {
elements[id].value = text; elements[id].value = value;
}; };
funcs.insert_text = function(id, text) { funcs.insert_text = function(id, text) {

View File

@ -232,8 +232,8 @@ class MainWindow(QWidget):
width = self.width() - 2 * padding width = self.width() - 2 * padding
left = padding left = padding
else: else:
width = size_hint.width() width = min(size_hint.width(), self.width() - 2 * padding)
left = (self.width() - size_hint.width()) / 2 if centered else 0 left = (self.width() - width) / 2 if centered else 0
height_padding = 20 height_padding = 20
status_position = config.get('ui', 'status-position') status_position = config.get('ui', 'status-position')

View File

@ -149,6 +149,7 @@ class TabWidget(QTabWidget):
fields['title'] = page_title fields['title'] = page_title
fields['title_sep'] = ' - ' if page_title else '' fields['title_sep'] = ' - ' if page_title else ''
fields['perc_raw'] = tab.progress() fields['perc_raw'] = tab.progress()
fields['backend'] = objreg.get('args').backend
if tab.load_status() == usertypes.LoadStatus.loading: if tab.load_status() == usertypes.LoadStatus.loading:
fields['perc'] = '[{}%] '.format(tab.progress()) fields['perc'] = '[{}%] '.format(tab.progress())

View File

@ -72,10 +72,7 @@ class CrashHandler(QObject):
def handle_segfault(self): def handle_segfault(self):
"""Handle a segfault from a previous run.""" """Handle a segfault from a previous run."""
data_dir = standarddir.data() logname = os.path.join(standarddir.data(), 'crash.log')
if data_dir is None:
return
logname = os.path.join(data_dir, 'crash.log')
try: try:
# First check if an old logfile exists. # First check if an old logfile exists.
if os.path.exists(logname): if os.path.exists(logname):
@ -131,10 +128,7 @@ class CrashHandler(QObject):
def _init_crashlogfile(self): def _init_crashlogfile(self):
"""Start a new logfile and redirect faulthandler to it.""" """Start a new logfile and redirect faulthandler to it."""
assert not self._args.no_err_windows assert not self._args.no_err_windows
data_dir = standarddir.data() logname = os.path.join(standarddir.data(), 'crash.log')
if data_dir is None:
return
logname = os.path.join(data_dir, 'crash.log')
try: try:
self._crash_log_file = open(logname, 'w', encoding='ascii') self._crash_log_file = open(logname, 'w', encoding='ascii')
except OSError: except OSError:

View File

@ -57,9 +57,6 @@ class BaseLineParser(QObject):
""" """
super().__init__(parent) super().__init__(parent)
self._configdir = configdir self._configdir = configdir
if self._configdir is None:
self._configfile = None
else:
self._configfile = os.path.join(self._configdir, fname) self._configfile = os.path.join(self._configdir, fname)
self._fname = fname self._fname = fname
self._binary = binary self._binary = binary
@ -76,8 +73,6 @@ class BaseLineParser(QObject):
Return: Return:
True if the file should be saved, False otherwise. True if the file should be saved, False otherwise.
""" """
if self._configdir is None:
return False
if not os.path.exists(self._configdir): if not os.path.exists(self._configdir):
os.makedirs(self._configdir, 0o755) os.makedirs(self._configdir, 0o755)
return True return True
@ -222,7 +217,7 @@ class LineParser(BaseLineParser):
binary: Whether to open the file in binary mode. binary: Whether to open the file in binary mode.
""" """
super().__init__(configdir, fname, binary=binary, parent=parent) super().__init__(configdir, fname, binary=binary, parent=parent)
if configdir is None or not os.path.isfile(self._configfile): if not os.path.isfile(self._configfile):
self.data = [] self.data = []
else: else:
log.init.debug("Reading {}".format(self._configfile)) log.init.debug("Reading {}".format(self._configfile))

View File

@ -47,10 +47,6 @@ def init(parent=None):
Args: Args:
parent: The parent to use for the SessionManager. parent: The parent to use for the SessionManager.
""" """
data_dir = standarddir.data()
if data_dir is None:
base_path = None
else:
base_path = os.path.join(standarddir.data(), 'sessions') base_path = os.path.join(standarddir.data(), 'sessions')
try: try:
os.mkdir(base_path) os.mkdir(base_path)
@ -137,11 +133,6 @@ class SessionManager(QObject):
if os.path.isabs(path) and ((not check_exists) or if os.path.isabs(path) and ((not check_exists) or
os.path.exists(path)): os.path.exists(path)):
return path return path
elif self._base_path is None:
if check_exists:
raise SessionNotFoundError(name)
else:
return None
else: else:
path = os.path.join(self._base_path, name + '.yml') path = os.path.join(self._base_path, name + '.yml')
if check_exists and not os.path.exists(path): if check_exists and not os.path.exists(path):
@ -282,8 +273,6 @@ class SessionManager(QObject):
""" """
name = self._get_session_name(name) name = self._get_session_name(name)
path = self._get_session_path(name) path = self._get_session_path(name)
if path is None:
raise SessionError("No data storage configured.")
log.sessions.debug("Saving session {} to {}...".format(name, path)) log.sessions.debug("Saving session {} to {}...".format(name, path))
if last_window: if last_window:
@ -400,8 +389,6 @@ class SessionManager(QObject):
def list_sessions(self): def list_sessions(self):
"""Get a list of all session names.""" """Get a list of all session names."""
sessions = [] sessions = []
if self._base_path is None:
return sessions
for filename in os.listdir(self._base_path): for filename in os.listdir(self._base_path):
base, ext = os.path.splitext(filename) base, ext = os.path.splitext(filename)
if ext == '.yml': if ext == '.yml':

View File

@ -47,14 +47,7 @@ def get_argparser():
"""Get the argparse parser.""" """Get the argparse parser."""
parser = argparse.ArgumentParser(prog='qutebrowser', parser = argparse.ArgumentParser(prog='qutebrowser',
description=qutebrowser.__description__) description=qutebrowser.__description__)
parser.add_argument('-c', '--confdir', help="Set config directory (empty " parser.add_argument('--basedir', help="Base directory for all storage.")
"for no config storage).")
parser.add_argument('--datadir', help="Set data directory (empty for "
"no data storage).")
parser.add_argument('--cachedir', help="Set cache directory (empty for "
"no cache storage).")
parser.add_argument('--basedir', help="Base directory for all storage. "
"Other --*dir arguments are ignored if this is given.")
parser.add_argument('-V', '--version', help="Show version and quit.", parser.add_argument('-V', '--version', help="Show version and quit.",
action='store_true') action='store_true')
parser.add_argument('-s', '--set', help="Set a temporary setting for " parser.add_argument('-s', '--set', help="Set a temporary setting for "
@ -112,8 +105,10 @@ def get_argparser():
"temporary basedir.") "temporary basedir.")
debug.add_argument('--no-err-windows', action='store_true', help="Don't " debug.add_argument('--no-err-windows', action='store_true', help="Don't "
"show any error windows (used for tests/smoke.py).") "show any error windows (used for tests/smoke.py).")
debug.add_argument('--qt-arg', help="Pass an argument with a value to Qt.", debug.add_argument('--qt-arg', help="Pass an argument with a value to Qt. "
nargs=2) "For example, you can do "
"`--qt-arg geometry 650x555+200+300` to set the window "
"geometry.", nargs=2, metavar=('NAME', 'VALUE'))
debug.add_argument('--qt-flag', help="Pass an argument to Qt as flag.", debug.add_argument('--qt-flag', help="Pass an argument to Qt as flag.",
nargs=1) nargs=1)
parser.add_argument('command', nargs='*', help="Commands to execute on " parser.add_argument('command', nargs='*', help="Commands to execute on "
@ -124,6 +119,11 @@ def get_argparser():
return parser return parser
def directory(arg):
if not arg:
raise argparse.ArgumentTypeError("Invalid empty value")
def logfilter_error(logfilter: str): def logfilter_error(logfilter: str):
"""Validate logger names passed to --logfilter. """Validate logger names passed to --logfilter.

View File

@ -23,7 +23,6 @@ import os
import os.path import os.path
import traceback import traceback
import mimetypes import mimetypes
import base64
import jinja2 import jinja2
import jinja2.exceptions import jinja2.exceptions
@ -82,8 +81,7 @@ def data_url(path):
filename = utils.resource_filename(path) filename = utils.resource_filename(path)
mimetype = mimetypes.guess_type(filename) mimetype = mimetypes.guess_type(filename)
assert mimetype is not None, path assert mimetype is not None, path
b64 = base64.b64encode(data).decode('ascii') return urlutils.data_url(mimetype[0], data).toString()
return 'data:{};charset=utf-8;base64,{}'.format(mimetype[0], b64)
def render(template, **kwargs): def render(template, **kwargs):

View File

@ -21,6 +21,7 @@
import os import os
import sys import sys
import shutil
import os.path import os.path
from PyQt5.QtCore import QCoreApplication, QStandardPaths from PyQt5.QtCore import QCoreApplication, QStandardPaths
@ -43,7 +44,7 @@ def config():
# WORKAROUND - see # WORKAROUND - see
# https://bugreports.qt.io/browse/QTBUG-38872 # https://bugreports.qt.io/browse/QTBUG-38872
path = os.path.join(path, appname) path = os.path.join(path, appname)
_maybe_create(path) _create(path)
return path return path
@ -61,7 +62,7 @@ def data():
QStandardPaths.ConfigLocation) QStandardPaths.ConfigLocation)
if data_path == config_path: if data_path == config_path:
path = os.path.join(path, 'data') path = os.path.join(path, 'data')
_maybe_create(path) _create(path)
return path return path
@ -82,7 +83,7 @@ def cache():
overridden, path = _from_args(typ, _args) overridden, path = _from_args(typ, _args)
if not overridden: if not overridden:
path = _writable_location(typ) path = _writable_location(typ)
_maybe_create(path) _create(path)
return path return path
@ -92,7 +93,7 @@ def download():
overridden, path = _from_args(typ, _args) overridden, path = _from_args(typ, _args)
if not overridden: if not overridden:
path = _writable_location(typ) path = _writable_location(typ)
_maybe_create(path) _create(path)
return path return path
@ -116,7 +117,7 @@ def runtime():
# maximum length of 104 chars), so we don't add the username here... # maximum length of 104 chars), so we don't add the username here...
appname = QCoreApplication.instance().applicationName() appname = QCoreApplication.instance().applicationName()
path = os.path.join(path, appname) path = os.path.join(path, appname)
_maybe_create(path) _create(path)
return path return path
@ -145,11 +146,6 @@ def _from_args(typ, args):
override: boolean, if the user did override the path override: boolean, if the user did override the path
path: The overridden path, or None to turn off storage. path: The overridden path, or None to turn off storage.
""" """
typ_to_argparse_arg = {
QStandardPaths.ConfigLocation: 'confdir',
QStandardPaths.DataLocation: 'datadir',
QStandardPaths.CacheLocation: 'cachedir',
}
basedir_suffix = { basedir_suffix = {
QStandardPaths.ConfigLocation: 'config', QStandardPaths.ConfigLocation: 'config',
QStandardPaths.DataLocation: 'data', QStandardPaths.DataLocation: 'data',
@ -158,9 +154,6 @@ def _from_args(typ, args):
QStandardPaths.RuntimeLocation: 'runtime', QStandardPaths.RuntimeLocation: 'runtime',
} }
if args is None:
return (False, None)
if getattr(args, 'basedir', None) is not None: if getattr(args, 'basedir', None) is not None:
basedir = args.basedir basedir = args.basedir
@ -169,22 +162,12 @@ def _from_args(typ, args):
except KeyError: # pragma: no cover except KeyError: # pragma: no cover
return (False, None) return (False, None)
return (True, os.path.abspath(os.path.join(basedir, suffix))) return (True, os.path.abspath(os.path.join(basedir, suffix)))
try:
argname = typ_to_argparse_arg[typ]
except KeyError:
return (False, None)
arg_value = getattr(args, argname)
if arg_value is None:
return (False, None)
elif arg_value == '':
return (True, None)
else: else:
return (True, arg_value) return (False, None)
def _maybe_create(path): def _create(path):
"""Create the `path` directory if path is not None. """Create the `path` directory.
From the XDG basedir spec: From the XDG basedir spec:
If, when attempting to write a file, the destination directory is If, when attempting to write a file, the destination directory is
@ -192,7 +175,6 @@ def _maybe_create(path):
0700. If the destination directory exists already the permissions 0700. If the destination directory exists already the permissions
should not be changed. should not be changed.
""" """
if path is not None:
try: try:
os.makedirs(path, 0o700) os.makedirs(path, 0o700)
except FileExistsError: except FileExistsError:
@ -207,6 +189,8 @@ def init(args):
log.init.debug("Base directory: {}".format(args.basedir)) log.init.debug("Base directory: {}".format(args.basedir))
_args = args _args = args
_init_cachedir_tag() _init_cachedir_tag()
if args is not None:
_move_webengine_data()
def _init_cachedir_tag(): def _init_cachedir_tag():
@ -214,10 +198,7 @@ def _init_cachedir_tag():
See http://www.brynosaurus.com/cachedir/spec.html See http://www.brynosaurus.com/cachedir/spec.html
""" """
cache_dir = cache() cachedir_tag = os.path.join(cache(), 'CACHEDIR.TAG')
if cache_dir is None:
return
cachedir_tag = os.path.join(cache_dir, 'CACHEDIR.TAG')
if not os.path.exists(cachedir_tag): if not os.path.exists(cachedir_tag):
try: try:
with open(cachedir_tag, 'w', encoding='utf-8') as f: with open(cachedir_tag, 'w', encoding='utf-8') as f:
@ -229,3 +210,49 @@ def _init_cachedir_tag():
"cachedir/\n") "cachedir/\n")
except OSError: except OSError:
log.init.exception("Failed to create CACHEDIR.TAG") log.init.exception("Failed to create CACHEDIR.TAG")
def _move_webengine_data():
"""Move QtWebEngine data from an older location to the new one."""
# Do NOT use _writable_location here as that'd give us a wrong path
old_data_dir = QStandardPaths.writableLocation(QStandardPaths.DataLocation)
old_cache_dir = QStandardPaths.writableLocation(
QStandardPaths.CacheLocation)
new_data_dir = os.path.join(data(), 'webengine')
new_cache_dir = os.path.join(cache(), 'webengine')
if (not os.path.exists(os.path.join(old_data_dir, 'QtWebEngine')) and
not os.path.exists(os.path.join(old_cache_dir, 'QtWebEngine'))):
return
log.init.debug("Moving QtWebEngine data from {} to {}".format(
old_data_dir, new_data_dir))
log.init.debug("Moving QtWebEngine cache from {} to {}".format(
old_cache_dir, new_cache_dir))
if os.path.exists(new_data_dir):
log.init.warning("Failed to move old QtWebEngine data as {} already "
"exists!".format(new_data_dir))
return
if os.path.exists(new_cache_dir):
log.init.warning("Failed to move old QtWebEngine cache as {} already "
"exists!".format(new_cache_dir))
return
try:
shutil.move(os.path.join(old_data_dir, 'QtWebEngine', 'Default'),
new_data_dir)
shutil.move(os.path.join(old_cache_dir, 'QtWebEngine', 'Default'),
new_cache_dir)
# Remove e.g.
# ~/.local/share/qutebrowser/qutebrowser/QtWebEngine/Default
if old_data_dir.split(os.sep)[-2:] == ['qutebrowser', 'qutebrowser']:
log.init.debug("Removing {} / {}".format(
old_data_dir, old_cache_dir))
for old_dir in old_data_dir, old_cache_dir:
os.rmdir(os.path.join(old_dir, 'QtWebEngine'))
os.rmdir(old_dir)
except OSError as e:
log.init.exception("Failed to move old QtWebEngine data/cache: "
"{}".format(e))

View File

@ -20,6 +20,7 @@
"""Utils regarding URL handling.""" """Utils regarding URL handling."""
import re import re
import base64
import os.path import os.path
import ipaddress import ipaddress
import posixpath import posixpath
@ -580,3 +581,11 @@ def file_url(path):
path: The absolute path to the local file path: The absolute path to the local file
""" """
return QUrl.fromLocalFile(path).toString(QUrl.FullyEncoded) return QUrl.fromLocalFile(path).toString(QUrl.FullyEncoded)
def data_url(mimetype, data):
"""Get a data: QUrl for the given data."""
b64 = base64.b64encode(data).decode('ascii')
url = QUrl('data:{};base64,{}'.format(mimetype, b64))
qtutils.ensure_valid(url)
return url

View File

@ -132,14 +132,7 @@ if not getattr(sys, 'frozen', False):
def pytest_collection_modifyitems(config, items): def pytest_collection_modifyitems(config, items):
"""Apply @qtwebengine_* markers; skip unittests with QUTE_BDD_WEBENGINE.""" """Apply @qtwebengine_* markers; skip unittests with QUTE_BDD_WEBENGINE."""
vercheck = qtutils.version_check
qtbug_54419_fixed = ((vercheck('5.6.2') and not vercheck('5.7.0')) or
qtutils.version_check('5.7.1') or
os.environ.get('QUTE_QTBUG54419_PATCHED', ''))
markers = [ markers = [
('qtwebengine_createWindow', 'Skipped because of QTBUG-54419',
pytest.mark.skipif, not qtbug_54419_fixed and config.webengine),
('qtwebengine_todo', 'QtWebEngine TODO', pytest.mark.xfail, ('qtwebengine_todo', 'QtWebEngine TODO', pytest.mark.xfail,
config.webengine), config.webengine),
('qtwebengine_skip', 'Skipped with QtWebEngine', pytest.mark.skipif, ('qtwebengine_skip', 'Skipped with QtWebEngine', pytest.mark.skipif,

View File

@ -121,6 +121,8 @@ def set_setting_given(quteproc, httpbin, sect, opt, value):
This is available as "Given:" step so it can be used as "Background:". This is available as "Given:" step so it can be used as "Background:".
""" """
if value == '<empty>':
value = ''
value = value.replace('(port)', str(httpbin.port)) value = value.replace('(port)', str(httpbin.port))
quteproc.set_setting(sect, opt, value) quteproc.set_setting(sect, opt, value)
@ -211,6 +213,8 @@ def open_path(quteproc, path):
@bdd.when(bdd.parsers.parse("I set {sect} -> {opt} to {value}")) @bdd.when(bdd.parsers.parse("I set {sect} -> {opt} to {value}"))
def set_setting(quteproc, httpbin, sect, opt, value): def set_setting(quteproc, httpbin, sect, opt, value):
"""Set a qutebrowser setting.""" """Set a qutebrowser setting."""
if value == '<empty>':
value = ''
value = value.replace('(port)', str(httpbin.port)) value = value.replace('(port)', str(httpbin.port))
quteproc.set_setting(sect, opt, value) quteproc.set_setting(sect, opt, value)
@ -479,7 +483,7 @@ def check_header(quteproc, header, value):
content = quteproc.get_content() content = quteproc.get_content()
data = json.loads(content) data = json.loads(content)
print(data) print(data)
assert data['headers'][header] == value assert utils.pattern_match(pattern=value, value=data['headers'][header])
@bdd.then(bdd.parsers.parse('the page should contain the html "{text}"')) @bdd.then(bdd.parsers.parse('the page should contain the html "{text}"'))

View File

@ -93,3 +93,15 @@ Feature: Opening external editors
And I wait for "Read back: foobar" in the log And I wait for "Read back: foobar" in the log
And I run :click-element id qute-button And I run :click-element id qute-button
Then the javascript message "text: foobar" should be logged Then the javascript message "text: foobar" should be logged
Scenario: Spawning an editor with existing text
When I set up a fake editor replacing "foo" by "bar"
And I open data/editor.html
And I run :click-element id qute-textarea
And I wait for "Clicked editable element!" in the log
And I run :insert-text foo
And I wait for "Inserting text into element *" in the log
And I run :open-editor
And I wait for "Read back: bar" in the log
And I run :click-element id qute-button
Then the javascript message "text: bar" should be logged

View File

@ -22,7 +22,6 @@ Feature: Using hints
### Opening in current or new tab ### Opening in current or new tab
@qtwebengine_createWindow
Scenario: Following a hint and force to open in current tab. Scenario: Following a hint and force to open in current tab.
When I open data/hints/link_blank.html When I open data/hints/link_blank.html
And I hint with args "links current" and follow a And I hint with args "links current" and follow a
@ -30,7 +29,7 @@ Feature: Using hints
Then the following tabs should be open: Then the following tabs should be open:
- data/hello.txt (active) - data/hello.txt (active)
@qtwebengine_createWindow @qtwebengine_skip: Opens in background
Scenario: Following a hint and allow to open in new tab. Scenario: Following a hint and allow to open in new tab.
When I open data/hints/link_blank.html When I open data/hints/link_blank.html
And I hint with args "links normal" and follow a And I hint with args "links normal" and follow a
@ -39,7 +38,6 @@ Feature: Using hints
- data/hints/link_blank.html - data/hints/link_blank.html
- data/hello.txt (active) - data/hello.txt (active)
@qtwebengine_createWindow
Scenario: Following a hint to link with sub-element and force to open in current tab. Scenario: Following a hint to link with sub-element and force to open in current tab.
When I open data/hints/link_span.html When I open data/hints/link_span.html
And I run :tab-close And I run :tab-close
@ -186,13 +184,12 @@ Feature: Using hints
And I hint wht args "links normal" and follow a And I hint wht args "links normal" and follow a
Then "navigation request: url http://localhost:*/data/hello2.txt, type NavigationTypeLinkClicked, *" should be logged Then "navigation request: url http://localhost:*/data/hello2.txt, type NavigationTypeLinkClicked, *" should be logged
@qtwebengine_createWindow @qtwebengine_skip: Opens in new tab due to Chromium bug
Scenario: Opening a link inside a specific iframe Scenario: Opening a link inside a specific iframe
When I open data/hints/iframe_target.html When I open data/hints/iframe_target.html
And I hint with args "links normal" and follow a And I hint with args "links normal" and follow a
Then "navigation request: url http://localhost:*/data/hello.txt, type NavigationTypeLinkClicked, *" should be logged Then "navigation request: url http://localhost:*/data/hello.txt, type NavigationTypeLinkClicked, *" should be logged
@qtwebengine_createWindow
Scenario: Opening a link with specific target frame in a new tab Scenario: Opening a link with specific target frame in a new tab
When I open data/hints/iframe_target.html When I open data/hints/iframe_target.html
And I run :tab-only And I run :tab-only

View File

@ -48,7 +48,6 @@ Feature: Page history
Then the history file should contain: Then the history file should contain:
http://localhost:(port)/status/404 Error loading page: http://localhost:(port)/status/404 http://localhost:(port)/status/404 Error loading page: http://localhost:(port)/status/404
@qtwebengine_createWindow
Scenario: History with invalid URL Scenario: History with invalid URL
When I open data/javascript/window_open.html When I open data/javascript/window_open.html
And I run :click-element id open-invalid And I run :click-element id open-invalid

View File

@ -7,8 +7,6 @@ Feature: Javascript stuff
And I open data/javascript/consolelog.html And I open data/javascript/consolelog.html
Then the javascript message "console.log works!" should be logged Then the javascript message "console.log works!" should be logged
# Causes segfaults...
@qtwebengine_createWindow
Scenario: Opening/Closing a window via JS Scenario: Opening/Closing a window via JS
When I open data/javascript/window_open.html When I open data/javascript/window_open.html
And I run :tab-only And I run :tab-only
@ -18,8 +16,6 @@ Feature: Javascript stuff
And I run :click-element id close-normal And I run :click-element id close-normal
Then "Focus object changed: *" should be logged Then "Focus object changed: *" should be logged
# Causes segfaults...
@qtwebengine_createWindow
Scenario: Opening/closing a modal window via JS Scenario: Opening/closing a modal window via JS
When I open data/javascript/window_open.html When I open data/javascript/window_open.html
And I run :tab-only And I run :tab-only
@ -37,14 +33,14 @@ Feature: Javascript stuff
Scenario: Closing a JS window twice (issue 906) - qtwebkit Scenario: Closing a JS window twice (issue 906) - qtwebkit
When I open about:blank When I open about:blank
And I run :tab-only And I run :tab-only
When I open data/javascript/window_open.html in a new tab And I open data/javascript/window_open.html in a new tab
And I run :click-element id open-normal And I run :click-element id open-normal
And I wait for "Changing title for idx 2 to 'about:blank'" in the log And I wait for "Changing title for idx 2 to 'about:blank'" in the log
And I run :tab-focus 2 And I run :tab-focus 2
And I run :click-element id close-twice And I run :click-element id close-twice
Then "Requested to close * which does not exist!" should be logged Then "Requested to close * which does not exist!" should be logged
@qtwebengine_createWindow @qtwebkit_skip @qtwebkit_skip @flaky
Scenario: Closing a JS window twice (issue 906) - qtwebengine Scenario: Closing a JS window twice (issue 906) - qtwebengine
When I open about:blank When I open about:blank
And I run :tab-only And I run :tab-only
@ -56,7 +52,6 @@ Feature: Javascript stuff
And I wait for "Focus object changed: *" in the log And I wait for "Focus object changed: *" in the log
Then no crash should happen Then no crash should happen
@qtwebengine_createWindow
Scenario: Opening window without user interaction with javascript-can-open-windows-automatically set to true Scenario: Opening window without user interaction with javascript-can-open-windows-automatically set to true
When I open data/hello.txt When I open data/hello.txt
And I set content -> javascript-can-open-windows-automatically to true And I set content -> javascript-can-open-windows-automatically to true
@ -64,7 +59,6 @@ Feature: Javascript stuff
And I run :jseval if (window.open('about:blank')) { console.log('window opened'); } else { console.log('error while opening window'); } And I run :jseval if (window.open('about:blank')) { console.log('window opened'); } else { console.log('error while opening window'); }
Then the javascript message "window opened" should be logged Then the javascript message "window opened" should be logged
@qtwebengine_createWindow
Scenario: Opening window without user interaction with javascript-can-open-windows-automatically set to false Scenario: Opening window without user interaction with javascript-can-open-windows-automatically set to false
When I open data/hello.txt When I open data/hello.txt
And I set content -> javascript-can-open-windows-automatically to false And I set content -> javascript-can-open-windows-automatically to false

View File

@ -415,6 +415,16 @@ Feature: Various utility commands.
And I open headers And I open headers
Then the header Accept-Language should be set to en,de Then the header Accept-Language should be set to en,de
Scenario: Setting a custom user-agent header
When I set network -> user-agent to toaster
And I open headers
Then the header User-Agent should be set to toaster
Scenario: Setting the default user-agent header
When I set network -> user-agent to <empty>
And I open headers
Then the header User-Agent should be set to Mozilla/5.0 *
## :messages ## :messages
Scenario: Showing error messages Scenario: Showing error messages
@ -556,7 +566,7 @@ Feature: Various utility commands.
Scenario: Clicking an element with unknown ID Scenario: Clicking an element with unknown ID
When I open data/click_element.html When I open data/click_element.html
And I run :click-element id blah And I run :click-element id blah
Then the error "No element found!" should be shown Then the error "No element found with id blah!" should be shown
Scenario: Clicking an element by ID Scenario: Clicking an element by ID
When I open data/click_element.html When I open data/click_element.html

View File

@ -197,6 +197,14 @@ Feature: Prompts
And I run :prompt-accept no And I run :prompt-accept no
Then a SSL error page should be shown Then a SSL error page should be shown
Scenario: SSL error with ssl-strict = ask -> abort
When I clear SSL errors
And I set network -> ssl-strict to ask
And I load an SSL page
And I wait for a prompt
And I run :leave-mode
Then a SSL error page should be shown
# Geolocation # Geolocation
Scenario: Always rejecting geolocation Scenario: Always rejecting geolocation

View File

@ -124,11 +124,10 @@ Feature: Miscellaneous utility commands exposed to the user.
Then the page should not be scrolled Then the page should not be scrolled
And the error "prompt-accept: This command is only allowed in prompt/yesno mode, not normal." should be shown And the error "prompt-accept: This command is only allowed in prompt/yesno mode, not normal." should be shown
@qtwebengine_createWindow
Scenario: :repeat-command with mode-switching command Scenario: :repeat-command with mode-switching command
When I open data/hints/link_blank.html When I open data/hints/link_blank.html
And I run :tab-only And I run :tab-only
And I hint with args "all" And I hint with args "all tab-fg"
And I run :leave-mode And I run :leave-mode
And I run :repeat-command And I run :repeat-command
And I run :follow-hint a And I run :follow-hint a

View File

@ -68,24 +68,6 @@ def temp_basedir_env(tmpdir, short_tmpdir):
return env return env
@pytest.mark.linux
def test_no_config(request, temp_basedir_env, quteproc_new):
"""Test starting with -c ""."""
args = ['-c', ''] + _base_args(request.config)
quteproc_new.start(args, env=temp_basedir_env)
quteproc_new.send_cmd(':quit')
quteproc_new.wait_for_quit()
@pytest.mark.linux
def test_no_cache(request, temp_basedir_env, quteproc_new):
"""Test starting with --cachedir=""."""
args = ['--cachedir='] + _base_args(request.config)
quteproc_new.start(args, env=temp_basedir_env)
quteproc_new.send_cmd(':quit')
quteproc_new.wait_for_quit()
@pytest.mark.linux @pytest.mark.linux
def test_ascii_locale(request, httpbin, tmpdir, quteproc_new): def test_ascii_locale(request, httpbin, tmpdir, quteproc_new):
"""Test downloads with LC_ALL=C set. """Test downloads with LC_ALL=C set.

View File

@ -155,10 +155,11 @@ def tab_registry(win_registry):
@pytest.fixture @pytest.fixture
def fake_web_tab(stubs, tab_registry, mode_manager, qapp): def fake_web_tab(stubs, tab_registry, mode_manager, qapp, fake_args):
"""Fixture providing the FakeWebTab *class*.""" """Fixture providing the FakeWebTab *class*."""
if PYQT_VERSION < 0x050600: if PYQT_VERSION < 0x050600:
pytest.skip('Causes segfaults, see #1638') pytest.skip('Causes segfaults, see #1638')
fake_args.backend = 'webengine'
return stubs.FakeWebTab return stubs.FakeWebTab

View File

@ -29,7 +29,6 @@ from PyQt5.QtCore import pyqtSignal, QUrl, QObject
from qutebrowser.browser import adblock from qutebrowser.browser import adblock
from qutebrowser.utils import objreg from qutebrowser.utils import objreg
from qutebrowser.commands import cmdexc
pytestmark = pytest.mark.usefixtures('qapp', 'config_tmpdir') pytestmark = pytest.mark.usefixtures('qapp', 'config_tmpdir')
@ -225,32 +224,6 @@ def generic_blocklists(directory):
return [blocklist1, blocklist2, blocklist3, blocklist4, blocklist5] return [blocklist1, blocklist2, blocklist3, blocklist4, blocklist5]
def test_without_datadir(config_stub, tmpdir, monkeypatch, win_registry):
"""No directory for data configured so no hosts file exists.
Ensure CommandError is raised and no URL is blocked.
"""
config_stub.data = {
'content': {
'host-block-lists': generic_blocklists(tmpdir),
'host-blocking-enabled': True,
}
}
monkeypatch.setattr('qutebrowser.utils.standarddir.data', lambda: None)
host_blocker = adblock.HostBlocker()
with pytest.raises(cmdexc.CommandError) as excinfo:
host_blocker.adblock_update()
assert str(excinfo.value) == "No data storage is configured!"
host_blocker.read_hosts()
for str_url in URLS_TO_CHECK:
assert not host_blocker.is_blocked(QUrl(str_url))
# To test on_config_changed
config_stub.set('content', 'host-block-lists', None)
def test_disabled_blocking_update(basedir, config_stub, download_stub, def test_disabled_blocking_update(basedir, config_stub, download_stub,
data_tmpdir, tmpdir, win_registry, caplog): data_tmpdir, tmpdir, win_registry, caplog):
"""Ensure no URL is blocked when host blocking is disabled.""" """Ensure no URL is blocked when host blocking is disabled."""

View File

@ -111,16 +111,6 @@ def test_cache_size_deactivated(config_stub, tmpdir):
assert disk_cache.cacheSize() == 0 assert disk_cache.cacheSize() == 0
def test_cache_no_cache_dir(config_stub):
"""Confirm that the cache is deactivated when cache_dir is None."""
config_stub.data = {
'storage': {'cache-size': 1024},
'general': {'private-browsing': False},
}
disk_cache = cache.DiskCache(None)
assert disk_cache.cacheSize() == 0
def test_cache_existing_metadata_file(config_stub, tmpdir): def test_cache_existing_metadata_file(config_stub, tmpdir):
"""Test querying existing meta data file from activated cache.""" """Test querying existing meta data file from activated cache."""
config_stub.data = { config_stub.data = {

View File

@ -19,7 +19,6 @@
"""Tests for the global page history.""" """Tests for the global page history."""
import base64
import logging import logging
import pytest import pytest
@ -28,7 +27,7 @@ from hypothesis import strategies
from PyQt5.QtCore import QUrl from PyQt5.QtCore import QUrl
from qutebrowser.browser import history from qutebrowser.browser import history
from qutebrowser.utils import objreg from qutebrowser.utils import objreg, urlutils
class FakeWebHistory: class FakeWebHistory:
@ -65,13 +64,6 @@ def test_async_read_twice(monkeypatch, qtbot, tmpdir, caplog):
assert caplog.records[0].msg == expected assert caplog.records[0].msg == expected
def test_async_read_no_datadir(qtbot, config_stub, fake_save_manager):
config_stub.data = {'general': {'private-browsing': False}}
hist = history.WebHistory(hist_dir=None, hist_name='history')
with qtbot.waitSignal(hist.async_read_done):
list(hist.async_read())
@pytest.mark.parametrize('redirect', [True, False]) @pytest.mark.parametrize('redirect', [True, False])
def test_adding_item_during_async_read(qtbot, hist, redirect): def test_adding_item_during_async_read(qtbot, hist, redirect):
"""Check what happens when adding URL while reading the history.""" """Check what happens when adding URL while reading the history."""
@ -382,9 +374,8 @@ def hist_interface():
def test_history_interface(qtbot, webview, hist_interface): def test_history_interface(qtbot, webview, hist_interface):
html = "<a href='about:blank'>foo</a>" html = b"<a href='about:blank'>foo</a>"
data = base64.b64encode(html.encode('utf-8')).decode('ascii') url = urlutils.data_url('text/html', html)
url = QUrl("data:text/html;charset=utf-8;base64,{}".format(data))
with qtbot.waitSignal(webview.loadFinished): with qtbot.waitSignal(webview.loadFinished):
webview.load(url) webview.load(url)

View File

@ -258,8 +258,8 @@ class TestWebKitElement:
lambda e: e.has_frame(), lambda e: e.has_frame(),
lambda e: e.geometry(), lambda e: e.geometry(),
lambda e: e.style_property('visibility', strategy='computed'), lambda e: e.style_property('visibility', strategy='computed'),
lambda e: e.text(), lambda e: e.value(),
lambda e: e.set_text('foo'), lambda e: e.set_value('foo'),
lambda e: e.insert_text('foo'), lambda e: e.insert_text('foo'),
lambda e: e.is_writable(), lambda e: e.is_writable(),
lambda e: e.is_content_editable(), lambda e: e.is_content_editable(),
@ -271,7 +271,7 @@ class TestWebKitElement:
lambda e: e.rect_on_view(), lambda e: e.rect_on_view(),
lambda e: e._is_visible(None), lambda e: e._is_visible(None),
], ids=['str', 'getitem', 'setitem', 'delitem', 'contains', 'iter', 'len', ], ids=['str', 'getitem', 'setitem', 'delitem', 'contains', 'iter', 'len',
'frame', 'geometry', 'style_property', 'text', 'set_text', 'frame', 'geometry', 'style_property', 'value', 'set_value',
'insert_text', 'is_writable', 'is_content_editable', 'is_editable', 'insert_text', 'is_writable', 'is_content_editable', 'is_editable',
'is_text_input', 'remove_blank_target', 'outer_xml', 'tag_name', 'is_text_input', 'remove_blank_target', 'outer_xml', 'tag_name',
'rect_on_view', 'is_visible']) 'rect_on_view', 'is_visible'])
@ -411,28 +411,18 @@ class TestWebKitElement:
def test_style_property(self, elem): def test_style_property(self, elem):
assert elem.style_property('foo', strategy='computed') == 'bar' assert elem.style_property('foo', strategy='computed') == 'bar'
@pytest.mark.parametrize('use_js, editable, expected', [ def test_value(self, elem):
(True, 'false', 'js'),
(True, 'true', 'nojs'),
(False, 'false', 'nojs'),
(False, 'true', 'nojs'),
])
def test_text(self, use_js, editable, expected):
elem = get_webelem(attributes={'contenteditable': editable})
elem._elem.toPlainText.return_value = 'nojs'
elem._elem.evaluateJavaScript.return_value = 'js' elem._elem.evaluateJavaScript.return_value = 'js'
assert elem.text(use_js=use_js) == expected assert elem.value() == 'js'
@pytest.mark.parametrize('use_js, editable, text, uses_js, arg', [ @pytest.mark.parametrize('editable, value, uses_js, arg', [
(True, 'false', 'foo', True, "this.value='foo'"), ('false', 'foo', True, "this.value='foo'"),
(True, 'false', "foo'bar", True, r"this.value='foo\'bar'"), ('false', "foo'bar", True, r"this.value='foo\'bar'"),
(True, 'true', 'foo', False, 'foo'), ('true', 'foo', False, 'foo'),
(False, 'false', 'foo', False, 'foo'),
(False, 'true', 'foo', False, 'foo'),
]) ])
def test_set_text(self, use_js, editable, text, uses_js, arg): def test_set_value(self, editable, value, uses_js, arg):
elem = get_webelem(attributes={'contenteditable': editable}) elem = get_webelem(attributes={'contenteditable': editable})
elem.set_text(text, use_js=use_js) elem.set_value(value)
attr = 'evaluateJavaScript' if uses_js else 'setPlainText' attr = 'evaluateJavaScript' if uses_js else 'setPlainText'
called_mock = getattr(elem._elem, attr) called_mock = getattr(elem._elem, attr)
called_mock.assert_called_with(arg) called_mock.assert_called_with(arg)

View File

@ -23,9 +23,7 @@ import os.path
import configparser import configparser
import collections import collections
import shutil import shutil
from unittest import mock
from PyQt5.QtCore import QObject
from PyQt5.QtGui import QColor from PyQt5.QtGui import QColor
import pytest import pytest
@ -33,7 +31,6 @@ import qutebrowser
from qutebrowser.config import config, configexc, configdata from qutebrowser.config import config, configexc, configdata
from qutebrowser.config.parsers import keyconf from qutebrowser.config.parsers import keyconf
from qutebrowser.commands import runners from qutebrowser.commands import runners
from qutebrowser.utils import objreg, standarddir
class TestConfigParser: class TestConfigParser:
@ -43,12 +40,12 @@ class TestConfigParser:
Objects = collections.namedtuple('Objects', ['cp', 'cfg']) Objects = collections.namedtuple('Objects', ['cp', 'cfg'])
@pytest.fixture @pytest.fixture
def objects(self): def objects(self, tmpdir):
cp = configparser.ConfigParser(interpolation=None, cp = configparser.ConfigParser(interpolation=None,
comment_prefixes='#') comment_prefixes='#')
cp.optionxform = lambda opt: opt # be case-insensitive cp.optionxform = lambda opt: opt # be case-insensitive
cfg = config.ConfigManager() cfg = config.ConfigManager()
cfg.read(None, None) cfg.read(str(tmpdir), 'qutebrowser.conf')
return self.Objects(cp=cp, cfg=cfg) return self.Objects(cp=cp, cfg=cfg)
@pytest.mark.parametrize('config, section, option, value', [ @pytest.mark.parametrize('config, section, option, value', [
@ -247,7 +244,7 @@ class TestKeyConfigParser:
"""Test config.parsers.keyconf.KeyConfigParser.""" """Test config.parsers.keyconf.KeyConfigParser."""
def test_cmd_binding(self, cmdline_test, config_stub): def test_cmd_binding(self, cmdline_test, config_stub, tmpdir):
"""Test various command bindings. """Test various command bindings.
See https://github.com/The-Compiler/qutebrowser/issues/615 See https://github.com/The-Compiler/qutebrowser/issues/615
@ -256,7 +253,7 @@ class TestKeyConfigParser:
cmdline_test: A pytest fixture which provides testcases. cmdline_test: A pytest fixture which provides testcases.
""" """
config_stub.data = {'aliases': []} config_stub.data = {'aliases': []}
kcp = keyconf.KeyConfigParser(None, None) kcp = keyconf.KeyConfigParser(str(tmpdir), 'keys.conf')
kcp._cur_section = 'normal' kcp._cur_section = 'normal'
if cmdline_test.valid: if cmdline_test.valid:
kcp._read_command(cmdline_test.cmd) kcp._read_command(cmdline_test.cmd)
@ -379,17 +376,17 @@ class TestDefaultConfig:
"""Test validating of the default config.""" """Test validating of the default config."""
@pytest.mark.usefixtures('qapp') @pytest.mark.usefixtures('qapp')
def test_default_config(self): def test_default_config(self, tmpdir):
"""Test validating of the default config.""" """Test validating of the default config."""
conf = config.ConfigManager() conf = config.ConfigManager()
conf.read(None, None) conf.read(str(tmpdir), 'qutebrowser.conf')
conf._validate_all() conf._validate_all()
def test_default_key_config(self): def test_default_key_config(self, tmpdir):
"""Test validating of the default key config.""" """Test validating of the default key config."""
# We import qutebrowser.app so the cmdutils.register decorators run. # We import qutebrowser.app so the cmdutils.register decorators run.
import qutebrowser.app # pylint: disable=unused-variable import qutebrowser.app # pylint: disable=unused-variable
conf = keyconf.KeyConfigParser(None, None) conf = keyconf.KeyConfigParser(str(tmpdir), 'keys.conf')
runner = runners.CommandRunner(win_id=0) runner = runners.CommandRunner(win_id=0)
for sectname in configdata.KEY_DATA: for sectname in configdata.KEY_DATA:
for cmd in conf.get_bindings_for(sectname).values(): for cmd in conf.get_bindings_for(sectname).values():
@ -418,48 +415,3 @@ class TestDefaultConfig:
shutil.copy(full_path, str(tmpdir / 'qutebrowser.conf')) shutil.copy(full_path, str(tmpdir / 'qutebrowser.conf'))
conf = config.ConfigManager() conf = config.ConfigManager()
conf.read(str(tmpdir), 'qutebrowser.conf') conf.read(str(tmpdir), 'qutebrowser.conf')
@pytest.mark.integration
class TestConfigInit:
"""Test initializing of the config."""
@pytest.fixture(autouse=True)
def patch(self, fake_args):
objreg.register('app', QObject())
objreg.register('save-manager', mock.MagicMock())
fake_args.relaxed_config = False
old_standarddir_args = standarddir._args
yield
objreg.delete('app')
objreg.delete('save-manager')
# registered by config.init()
objreg.delete('config')
objreg.delete('key-config')
objreg.delete('state-config')
standarddir._args = old_standarddir_args
@pytest.fixture
def env(self, tmpdir):
conf_path = (tmpdir / 'config').ensure(dir=1)
data_path = (tmpdir / 'data').ensure(dir=1)
cache_path = (tmpdir / 'cache').ensure(dir=1)
env = {
'XDG_CONFIG_HOME': str(conf_path),
'XDG_DATA_HOME': str(data_path),
'XDG_CACHE_HOME': str(cache_path),
}
return env
def test_config_none(self, monkeypatch, env, fake_args):
"""Test initializing with config path set to None."""
fake_args.confdir = ''
fake_args.datadir = ''
fake_args.cachedir = ''
fake_args.basedir = None
for k, v in env.items():
monkeypatch.setenv(k, v)
standarddir.init(fake_args)
config.init()
assert not os.listdir(env['XDG_CONFIG_HOME'])

View File

@ -22,7 +22,6 @@ import re
import collections import collections
import itertools import itertools
import os.path import os.path
import base64
import warnings import warnings
import pytest import pytest
@ -1211,15 +1210,9 @@ def unrequired_class(**kwargs):
@pytest.mark.usefixtures('qapp') @pytest.mark.usefixtures('qapp')
@pytest.mark.usefixtures('config_tmpdir') @pytest.mark.usefixtures('config_tmpdir')
class TestFileAndUserStyleSheet: class TestFile:
"""Test File/UserStyleSheet.""" @pytest.fixture(params=[configtypes.File, unrequired_class])
@pytest.fixture(params=[
configtypes.File,
configtypes.UserStyleSheet,
unrequired_class,
])
def klass(self, request): def klass(self, request):
return request.param return request.param
@ -1227,18 +1220,12 @@ class TestFileAndUserStyleSheet:
def file_class(self): def file_class(self):
return configtypes.File return configtypes.File
@pytest.fixture
def userstylesheet_class(self):
return configtypes.UserStyleSheet
def _expected(self, klass, arg): def _expected(self, klass, arg):
"""Get the expected value.""" """Get the expected value."""
if not arg: if not arg:
return None return None
elif klass is configtypes.File: elif klass is configtypes.File:
return arg return arg
elif klass is configtypes.UserStyleSheet:
return QUrl.fromLocalFile(arg)
elif klass is unrequired_class: elif klass is unrequired_class:
return arg return arg
else: else:
@ -1262,11 +1249,6 @@ class TestFileAndUserStyleSheet:
os_mock.path.isfile.return_value = False os_mock.path.isfile.return_value = False
configtypes.File(required=False).validate('foobar') configtypes.File(required=False).validate('foobar')
def test_validate_does_not_exist_userstylesheet(self, os_mock):
"""Test validate with a file which does not exist (UserStyleSheet)."""
os_mock.path.isfile.return_value = False
configtypes.UserStyleSheet().validate('foobar')
def test_validate_exists_abs(self, klass, os_mock): def test_validate_exists_abs(self, klass, os_mock):
"""Test validate with a file which does exist.""" """Test validate with a file which does exist."""
os_mock.path.isfile.return_value = True os_mock.path.isfile.return_value = True
@ -1284,21 +1266,10 @@ class TestFileAndUserStyleSheet:
os_mock.path.join.assert_called_once_with( os_mock.path.join.assert_called_once_with(
'/home/foo/.config/', 'foobar') '/home/foo/.config/', 'foobar')
def test_validate_rel_config_none_file(self, os_mock, monkeypatch):
"""Test with a relative path and standarddir.config returning None."""
monkeypatch.setattr(
'qutebrowser.config.configtypes.standarddir.config', lambda: None)
os_mock.path.isabs.return_value = False
with pytest.raises(configexc.ValidationError):
configtypes.File().validate('foobar')
@pytest.mark.parametrize('configtype, value, raises', [ @pytest.mark.parametrize('configtype, value, raises', [
(configtypes.File(), 'foobar', True), (configtypes.File(), 'foobar', True),
(configtypes.UserStyleSheet(), 'foobar', False),
(configtypes.UserStyleSheet(), '\ud800', True),
(configtypes.File(required=False), 'foobar', False), (configtypes.File(required=False), 'foobar', False),
], ids=['file-foobar', 'userstylesheet-foobar', 'userstylesheet-unicode', ], ids=['file-foobar', 'file-optional-foobar'])
'file-optional-foobar'])
def test_validate_rel_inexistent(self, os_mock, monkeypatch, configtype, def test_validate_rel_inexistent(self, os_mock, monkeypatch, configtype,
value, raises): value, raises):
"""Test with a relative path and standarddir.config returning None.""" """Test with a relative path and standarddir.config returning None."""
@ -1347,7 +1318,6 @@ class TestFileAndUserStyleSheet:
def test_transform_relative(self, klass, os_mock, monkeypatch): def test_transform_relative(self, klass, os_mock, monkeypatch):
"""Test transform() with relative dir and an available configdir.""" """Test transform() with relative dir and an available configdir."""
os_mock.path.exists.return_value = True # for TestUserStyleSheet
os_mock.path.isabs.return_value = False os_mock.path.isabs.return_value = False
monkeypatch.setattr( monkeypatch.setattr(
'qutebrowser.config.configtypes.standarddir.config', 'qutebrowser.config.configtypes.standarddir.config',
@ -1355,18 +1325,6 @@ class TestFileAndUserStyleSheet:
expected = self._expected(klass, '/configdir/foo') expected = self._expected(klass, '/configdir/foo')
assert klass().transform('foo') == expected assert klass().transform('foo') == expected
@pytest.mark.parametrize('no_config', [False, True])
def test_transform_userstylesheet_base64(self, monkeypatch, no_config):
"""Test transform with a data string."""
if no_config:
monkeypatch.setattr(
'qutebrowser.config.configtypes.standarddir.config',
lambda: None)
b64 = base64.b64encode(b"test").decode('ascii')
url = QUrl("data:text/css;charset=utf-8;base64,{}".format(b64))
assert configtypes.UserStyleSheet().transform("test") == url
class TestDirectory: class TestDirectory:
@ -1990,10 +1948,11 @@ class TestUserAgent:
def test_validate_valid(self, klass, val): def test_validate_valid(self, klass, val):
klass(none_ok=True).validate(val) klass(none_ok=True).validate(val)
def test_validate_invalid(self, klass): @pytest.mark.parametrize('val', ['', 'überbrowser'])
def test_validate_invalid(self, klass, val):
"""Test validate with empty string and none_ok = False.""" """Test validate with empty string and none_ok = False."""
with pytest.raises(configexc.ValidationError): with pytest.raises(configexc.ValidationError):
klass().validate('') klass().validate(val)
def test_transform(self, klass): def test_transform(self, klass):
assert klass().transform('foobar') == 'foobar' assert klass().transform('foobar') == 'foobar'

View File

@ -50,8 +50,7 @@ def gen_classes():
@hypothesis.given(strategies.text()) @hypothesis.given(strategies.text())
@hypothesis.example('\x00') @hypothesis.example('\x00')
def test_configtypes_hypothesis(klass, s): def test_configtypes_hypothesis(klass, s):
if (klass in [configtypes.File, configtypes.UserStyleSheet] and if (klass == configtypes.File and sys.platform == 'linux' and
sys.platform == 'linux' and
not os.environ.get('DISPLAY', '')): not os.environ.get('DISPLAY', '')):
pytest.skip("No DISPLAY available") pytest.skip("No DISPLAY available")

View File

@ -52,15 +52,6 @@ class TestBaseLineParser:
lineparser._prepare_save() lineparser._prepare_save()
os_mock.makedirs.assert_called_with(self.CONFDIR, 0o755) os_mock.makedirs.assert_called_with(self.CONFDIR, 0o755)
def test_prepare_save_no_config(self, mocker):
"""Test if _prepare_save doesn't create a None config dir."""
os_mock = mocker.patch('qutebrowser.misc.lineparser.os')
os_mock.path.exists.return_value = True
lineparser = lineparsermod.BaseLineParser(None, self.FILENAME)
assert not lineparser._prepare_save()
assert not os_mock.makedirs.called
def test_double_open(self, mocker, lineparser): def test_double_open(self, mocker, lineparser):
"""Test if _open refuses reentry.""" """Test if _open refuses reentry."""
mocker.patch('builtins.open', mock.mock_open()) mocker.patch('builtins.open', mock.mock_open())
@ -158,15 +149,6 @@ class TestAppendLineParser:
lineparser.save() lineparser.save()
assert (tmpdir / 'file').read() == self._get_expected(new_data) assert (tmpdir / 'file').read() == self._get_expected(new_data)
def test_save_without_configdir(self, tmpdir, lineparser):
"""Test save() failing because no configdir was set."""
new_data = ['new data 1', 'new data 2']
lineparser.new_data = new_data
lineparser._configdir = None
assert not lineparser.save()
# make sure new data is still there
assert lineparser.new_data == new_data
def test_clear(self, tmpdir, lineparser): def test_clear(self, tmpdir, lineparser):
"""Check if calling clear() empties both pending and persisted data.""" """Check if calling clear() empties both pending and persisted data."""
lineparser.new_data = ['one', 'two'] lineparser.new_data = ['one', 'two']
@ -179,14 +161,6 @@ class TestAppendLineParser:
assert not lineparser.new_data assert not lineparser.new_data
assert (tmpdir / 'file').read() == "" assert (tmpdir / 'file').read() == ""
def test_clear_without_configdir(self, tmpdir, lineparser):
"""Test clear() failing because no configdir was set."""
new_data = ['new data 1', 'new data 2']
lineparser.new_data = new_data
lineparser._configdir = None
assert not lineparser.clear()
assert lineparser.new_data == new_data
def test_iter_without_open(self, lineparser): def test_iter_without_open(self, lineparser):
"""Test __iter__ without having called open().""" """Test __iter__ without having called open()."""
with pytest.raises(ValueError): with pytest.raises(ValueError):

View File

@ -40,9 +40,9 @@ webengine_refactoring_xfail = pytest.mark.xfail(
@pytest.fixture @pytest.fixture
def sess_man(): def sess_man(tmpdir):
"""Fixture providing a SessionManager with no session dir.""" """Fixture providing a SessionManager."""
return sessions.SessionManager(base_path=None) return sessions.SessionManager(base_path=str(tmpdir))
class TestInit: class TestInit:
@ -52,13 +52,6 @@ class TestInit:
yield yield
objreg.delete('session-manager') objreg.delete('session-manager')
def test_no_standarddir(self, monkeypatch):
monkeypatch.setattr('qutebrowser.misc.sessions.standarddir.data',
lambda: None)
sessions.init()
manager = objreg.get('session-manager')
assert manager._base_path is None
@pytest.mark.parametrize('create_dir', [True, False]) @pytest.mark.parametrize('create_dir', [True, False])
def test_with_standarddir(self, tmpdir, monkeypatch, create_dir): def test_with_standarddir(self, tmpdir, monkeypatch, create_dir):
monkeypatch.setattr('qutebrowser.misc.sessions.standarddir.data', monkeypatch.setattr('qutebrowser.misc.sessions.standarddir.data',
@ -110,16 +103,6 @@ class TestExists:
assert not man.exists(name) assert not man.exists(name)
@pytest.mark.parametrize('absolute', [True, False])
def test_no_datadir(self, sess_man, tmpdir, absolute):
abs_session = tmpdir / 'foo.yml'
abs_session.ensure()
if absolute:
assert sess_man.exists(str(abs_session))
else:
assert not sess_man.exists('foo')
@webengine_refactoring_xfail @webengine_refactoring_xfail
class TestSaveTab: class TestSaveTab:
@ -247,11 +230,6 @@ class TestSave:
objreg.delete('main-window', scope='window', window=0) objreg.delete('main-window', scope='window', window=0)
objreg.delete('tabbed-browser', scope='window', window=0) objreg.delete('tabbed-browser', scope='window', window=0)
def test_no_config_storage(self, sess_man):
with pytest.raises(sessions.SessionError) as excinfo:
sess_man.save('foo')
assert str(excinfo.value) == "No data storage configured."
def test_update_completion_signal(self, sess_man, tmpdir, qtbot): def test_update_completion_signal(self, sess_man, tmpdir, qtbot):
session_path = tmpdir / 'foo.yml' session_path = tmpdir / 'foo.yml'
with qtbot.waitSignal(sess_man.update_completion): with qtbot.waitSignal(sess_man.update_completion):
@ -415,9 +393,6 @@ def test_delete_update_completion_signal(sess_man, qtbot, tmpdir):
class TestListSessions: class TestListSessions:
def test_no_base_path(self, sess_man):
assert not sess_man.list_sessions()
def test_no_sessions(self, tmpdir): def test_no_sessions(self, tmpdir):
sess_man = sessions.SessionManager(str(tmpdir)) sess_man = sessions.SessionManager(str(tmpdir))
assert not sess_man.list_sessions() assert not sess_man.list_sessions()

View File

@ -102,7 +102,7 @@ def test_data_url():
print(data) print(data)
url = QUrl(data) url = QUrl(data)
assert url.isValid() assert url.isValid()
assert data == 'data:text/plain;charset=utf-8;base64,Zm9v' # 'foo' assert data == 'data:text/plain;base64,Zm9v' # 'foo'
def test_not_found(): def test_not_found():

View File

@ -110,6 +110,7 @@ class TestStandardDir:
(standarddir.data, 'XDG_DATA_HOME'), (standarddir.data, 'XDG_DATA_HOME'),
(standarddir.config, 'XDG_CONFIG_HOME'), (standarddir.config, 'XDG_CONFIG_HOME'),
(standarddir.cache, 'XDG_CACHE_HOME'), (standarddir.cache, 'XDG_CACHE_HOME'),
(standarddir.runtime, 'XDG_RUNTIME_DIR'),
]) ])
@pytest.mark.linux @pytest.mark.linux
def test_linux_explicit(self, monkeypatch, tmpdir, func, varname): def test_linux_explicit(self, monkeypatch, tmpdir, func, varname):
@ -163,57 +164,7 @@ DirArgTest = collections.namedtuple('DirArgTest', 'arg, expected')
@pytest.mark.usefixtures('reset_standarddir') @pytest.mark.usefixtures('reset_standarddir')
class TestArguments: class TestArguments:
"""Tests with confdir/cachedir/datadir arguments.""" """Tests the --basedir argument."""
@pytest.fixture(params=[DirArgTest('', None), DirArgTest('foo', 'foo')])
def testcase(self, request, tmpdir):
"""Fixture providing testcases."""
if request.param.expected is None:
return request.param
else:
# prepend tmpdir to both
arg = str(tmpdir / request.param.arg)
return DirArgTest(arg, arg)
def test_confdir(self, testcase):
"""Test --confdir."""
args = types.SimpleNamespace(confdir=testcase.arg, cachedir=None,
datadir=None, basedir=None)
standarddir.init(args)
assert standarddir.config() == testcase.expected
def test_cachedir(self, testcase):
"""Test --cachedir."""
args = types.SimpleNamespace(confdir=None, cachedir=testcase.arg,
datadir=None, basedir=None)
standarddir.init(args)
assert standarddir.cache() == testcase.expected
def test_datadir(self, testcase):
"""Test --datadir."""
args = types.SimpleNamespace(confdir=None, cachedir=None,
datadir=testcase.arg, basedir=None)
standarddir.init(args)
assert standarddir.data() == testcase.expected
def test_confdir_none(self, mocker):
"""Test --confdir with None given."""
# patch makedirs to a noop so we don't really create a directory
mocker.patch('qutebrowser.utils.standarddir.os.makedirs')
args = types.SimpleNamespace(confdir=None, cachedir=None, datadir=None,
basedir=None)
standarddir.init(args)
assert standarddir.config().split(os.sep)[-1] == 'qute_test'
def test_runtimedir(self, tmpdir, monkeypatch):
"""Test runtime dir (which has no args)."""
monkeypatch.setattr(
'qutebrowser.utils.standarddir.QStandardPaths.writableLocation',
lambda _typ: str(tmpdir))
args = types.SimpleNamespace(confdir=None, cachedir=None, datadir=None,
basedir=None)
standarddir.init(args)
assert standarddir.runtime() == str(tmpdir / 'qute_test')
@pytest.mark.parametrize('typ', ['config', 'data', 'cache', 'download', @pytest.mark.parametrize('typ', ['config', 'data', 'cache', 'download',
pytest.mark.linux('runtime')]) pytest.mark.linux('runtime')])
@ -239,13 +190,6 @@ class TestInitCacheDirTag:
"""Tests for _init_cachedir_tag.""" """Tests for _init_cachedir_tag."""
def test_no_cache_dir(self, mocker, monkeypatch):
"""Smoke test with cache() returning None."""
monkeypatch.setattr('qutebrowser.utils.standarddir.cache',
lambda: None)
mocker.patch('builtins.open', side_effect=AssertionError)
standarddir._init_cachedir_tag()
def test_existent_cache_dir_tag(self, tmpdir, mocker, monkeypatch): def test_existent_cache_dir_tag(self, tmpdir, mocker, monkeypatch):
"""Test with an existent CACHEDIR.TAG.""" """Test with an existent CACHEDIR.TAG."""
monkeypatch.setattr('qutebrowser.utils.standarddir.cache', monkeypatch.setattr('qutebrowser.utils.standarddir.cache',
@ -356,3 +300,102 @@ class TestSystemData:
standarddir.init(fake_args) standarddir.init(fake_args)
monkeypatch.setattr('sys.platform', "potato") monkeypatch.setattr('sys.platform', "potato")
assert standarddir.system_data() == standarddir.data() assert standarddir.system_data() == standarddir.data()
class TestMoveWebEngineData:
"""Test moving QtWebEngine data from an old location."""
@pytest.fixture(autouse=True)
def patch_standardpaths(self, tmpdir, monkeypatch):
locations = {
QStandardPaths.DataLocation: str(tmpdir / 'data'),
QStandardPaths.CacheLocation: str(tmpdir / 'cache'),
}
monkeypatch.setattr(standarddir.QStandardPaths, 'writableLocation',
locations.get)
monkeypatch.setattr(standarddir, 'data',
lambda: str(tmpdir / 'new_data'))
monkeypatch.setattr(standarddir, 'cache',
lambda: str(tmpdir / 'new_cache'))
@pytest.fixture
def files(self, tmpdir):
files = collections.namedtuple('Files', ['old_data', 'new_data',
'old_cache', 'new_cache'])
return files(
old_data=tmpdir / 'data' / 'QtWebEngine' / 'Default' / 'datafile',
new_data=tmpdir / 'new_data' / 'webengine' / 'datafile',
old_cache=(tmpdir / 'cache' / 'QtWebEngine' / 'Default' /
'cachefile'),
new_cache=(tmpdir / 'new_cache' / 'webengine' / 'cachefile'),
)
def test_no_webengine_dir(self, caplog):
"""Nothing should happen without any QtWebEngine directory."""
standarddir._move_webengine_data()
assert not any(rec.message.startswith('Moving QtWebEngine')
for rec in caplog.records)
def test_moving_data(self, files):
files.old_data.ensure()
files.old_cache.ensure()
standarddir._move_webengine_data()
assert not files.old_data.exists()
assert not files.old_cache.exists()
assert files.new_data.exists()
assert files.new_cache.exists()
@pytest.mark.parametrize('what', ['data', 'cache'])
def test_already_existing(self, files, caplog, what):
files.old_data.ensure()
files.old_cache.ensure()
if what == 'data':
files.new_data.ensure()
else:
files.new_cache.ensure()
with caplog.at_level(logging.WARNING):
standarddir._move_webengine_data()
record = caplog.records[-1]
expected = "Failed to move old QtWebEngine {}".format(what)
assert record.message.startswith(expected)
def test_deleting_empty_dirs(self, monkeypatch, tmpdir):
"""When we have a qutebrowser/qutebrowser subfolder, clean it up."""
old_data = tmpdir / 'data' / 'qutebrowser' / 'qutebrowser'
old_cache = tmpdir / 'cache' / 'qutebrowser' / 'qutebrowser'
locations = {
QStandardPaths.DataLocation: str(old_data),
QStandardPaths.CacheLocation: str(old_cache),
}
monkeypatch.setattr(standarddir.QStandardPaths, 'writableLocation',
locations.get)
old_data_file = old_data / 'QtWebEngine' / 'Default' / 'datafile'
old_cache_file = old_cache / 'QtWebEngine' / 'Default' / 'cachefile'
old_data_file.ensure()
old_cache_file.ensure()
standarddir._move_webengine_data()
assert not (tmpdir / 'data' / 'qutebrowser' / 'qutebrowser').exists()
assert not (tmpdir / 'cache' / 'qutebrowser' / 'qutebrowser').exists()
def test_deleting_error(self, files, monkeypatch, mocker, caplog):
"""When there was an error it should be logged."""
mock = mocker.Mock(side_effect=OSError('error'))
monkeypatch.setattr(standarddir.shutil, 'move', mock)
files.old_data.ensure()
files.old_cache.ensure()
with caplog.at_level(logging.ERROR):
standarddir._move_webengine_data()
record = caplog.records[-1]
expected = "Failed to move old QtWebEngine data/cache: error"
assert record.message == expected

View File

@ -731,3 +731,8 @@ class TestIncDecNumber:
def test_file_url(): def test_file_url():
assert urlutils.file_url('/foo/bar') == 'file:///foo/bar' assert urlutils.file_url('/foo/bar') == 'file:///foo/bar'
def test_data_url():
url = urlutils.data_url('text/plain', b'foo')
assert url == QUrl('data:text/plain;base64,Zm9v')