495 lines
19 KiB
Python
495 lines
19 KiB
Python
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
|
|
|
|
# Copyright 2014-2017 Florian Bruhin (The Compiler) <mail@qutebrowser.org>
|
|
#
|
|
# This file is part of qutebrowser.
|
|
#
|
|
# qutebrowser is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# qutebrowser is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with qutebrowser. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
"""Tests for qutebrowser.utils.standarddir."""
|
|
|
|
import os
|
|
import sys
|
|
import os.path
|
|
import types
|
|
import collections
|
|
import logging
|
|
import textwrap
|
|
|
|
from PyQt5.QtCore import QStandardPaths
|
|
import pytest
|
|
|
|
from qutebrowser.utils import standarddir
|
|
|
|
|
|
pytestmark = pytest.mark.usefixtures('qapp')
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
def clear_standarddir_cache(monkeypatch):
|
|
"""Make sure the standarddir cache is cleared before/after each test."""
|
|
monkeypatch.setattr(standarddir, '_locations', {})
|
|
yield
|
|
monkeypatch.setattr(standarddir, '_locations', {})
|
|
|
|
|
|
def test_fake_mac_auto_config(tmpdir, monkeypatch):
|
|
"""Test standardir.config(auto=True) on a fake Mac."""
|
|
monkeypatch.setattr(sys, 'platform', 'darwin')
|
|
monkeypatch.setenv('HOME', str(tmpdir))
|
|
expected = str(tmpdir) + '/.qutebrowser' # always with /
|
|
standarddir._init_config(args=None)
|
|
assert standarddir.config(auto=True) == expected
|
|
|
|
|
|
# FIXME:conf
|
|
# needs AppDataLocation
|
|
@pytest.mark.qt55
|
|
@pytest.mark.parametrize('what', ['data', 'config'])
|
|
def test_fake_windows_data_config(tmpdir, monkeypatch, what):
|
|
"""Make sure the config is correct on a fake Windows."""
|
|
monkeypatch.setattr(os, 'name', 'nt')
|
|
monkeypatch.setattr(standarddir.QStandardPaths, 'writableLocation',
|
|
lambda typ: str(tmpdir))
|
|
standarddir._init_config(args=None)
|
|
standarddir._init_data(args=None)
|
|
func = getattr(standarddir, what)
|
|
assert func() == str(tmpdir / what)
|
|
|
|
|
|
class TestWritableLocation:
|
|
|
|
"""Tests for _writable_location."""
|
|
|
|
def test_empty(self, monkeypatch):
|
|
"""Test QStandardPaths returning an empty value."""
|
|
monkeypatch.setattr(
|
|
'qutebrowser.utils.standarddir.QStandardPaths.writableLocation',
|
|
lambda typ: '')
|
|
with pytest.raises(standarddir.EmptyValueError):
|
|
standarddir._writable_location(QStandardPaths.DataLocation)
|
|
|
|
def test_sep(self, monkeypatch):
|
|
"""Make sure the right kind of separator is used."""
|
|
monkeypatch.setattr(standarddir.os, 'sep', '\\')
|
|
loc = standarddir._writable_location(QStandardPaths.DataLocation)
|
|
assert '/' not in loc
|
|
assert '\\' in loc
|
|
|
|
|
|
class TestStandardDir:
|
|
|
|
"""Tests for standarddir."""
|
|
|
|
@pytest.mark.parametrize('func, varname', [
|
|
(standarddir.data, 'XDG_DATA_HOME'),
|
|
(standarddir.config, 'XDG_CONFIG_HOME'),
|
|
(lambda: standarddir.config(auto=True), 'XDG_CONFIG_HOME'),
|
|
(standarddir.cache, 'XDG_CACHE_HOME'),
|
|
(standarddir.runtime, 'XDG_RUNTIME_DIR'),
|
|
])
|
|
@pytest.mark.linux
|
|
def test_linux_explicit(self, monkeypatch, tmpdir, func, varname):
|
|
"""Test dirs with XDG environment variables explicitly set.
|
|
|
|
Args:
|
|
func: The function to test.
|
|
varname: The environment variable which should be set.
|
|
"""
|
|
monkeypatch.setenv(varname, str(tmpdir))
|
|
standarddir._init_dirs()
|
|
assert func() == str(tmpdir / 'qute_test')
|
|
|
|
@pytest.mark.parametrize('func, subdirs', [
|
|
(standarddir.data, ['.local', 'share', 'qute_test']),
|
|
(standarddir.config, ['.config', 'qute_test']),
|
|
(lambda: standarddir.config(auto=True), ['.config', 'qute_test']),
|
|
(standarddir.cache, ['.cache', 'qute_test']),
|
|
(standarddir.download, ['Downloads']),
|
|
])
|
|
@pytest.mark.linux
|
|
def test_linux_normal(self, monkeypatch, tmpdir, func, subdirs):
|
|
"""Test dirs with XDG_*_HOME not set."""
|
|
monkeypatch.setenv('HOME', str(tmpdir))
|
|
for var in ['DATA', 'CONFIG', 'CACHE']:
|
|
monkeypatch.delenv('XDG_{}_HOME'.format(var), raising=False)
|
|
standarddir._init_dirs()
|
|
assert func() == str(tmpdir.join(*subdirs))
|
|
|
|
@pytest.mark.linux
|
|
@pytest.mark.qt_log_ignore(r'^QStandardPaths: ')
|
|
def test_linux_invalid_runtimedir(self, monkeypatch, tmpdir):
|
|
"""With invalid XDG_RUNTIME_DIR, fall back to TempLocation."""
|
|
monkeypatch.setenv('XDG_RUNTIME_DIR', str(tmpdir / 'does-not-exist'))
|
|
monkeypatch.setenv('TMPDIR', str(tmpdir / 'temp'))
|
|
standarddir._init_dirs()
|
|
assert standarddir.runtime() == str(tmpdir / 'temp' / 'qute_test')
|
|
|
|
def test_runtimedir_empty_tempdir(self, monkeypatch, tmpdir):
|
|
"""With an empty tempdir on non-Linux, we should raise."""
|
|
monkeypatch.setattr(standarddir.sys, 'platform', 'nt')
|
|
monkeypatch.setattr(standarddir.QStandardPaths, 'writableLocation',
|
|
lambda typ: '')
|
|
with pytest.raises(standarddir.EmptyValueError):
|
|
standarddir._init_runtime(args=None)
|
|
|
|
@pytest.mark.parametrize('func, elems, expected', [
|
|
(standarddir.data, 2, ['qute_test', 'data']),
|
|
(standarddir.config, 2, ['qute_test', 'config']),
|
|
(lambda: standarddir.config(auto=True), 2, ['qute_test', 'config']),
|
|
(standarddir.cache, 2, ['qute_test', 'cache']),
|
|
(standarddir.download, 1, ['Downloads']),
|
|
])
|
|
@pytest.mark.windows
|
|
def test_windows(self, func, elems, expected):
|
|
standarddir._init_dirs()
|
|
assert func().split(os.sep)[-elems:] == expected
|
|
|
|
@pytest.mark.parametrize('func, elems, expected', [
|
|
(standarddir.data, 2, ['Application Support', 'qute_test']),
|
|
(standarddir.config, 1, ['qute_test']),
|
|
(lambda: standarddir.config(auto=True), 2,
|
|
[os.path.expanduser('~'), '.qutebrowser']),
|
|
(standarddir.cache, 2, ['Caches', 'qute_test']),
|
|
(standarddir.download, 1, ['Downloads']),
|
|
])
|
|
@pytest.mark.mac
|
|
def test_mac(self, func, elems, expected):
|
|
standarddir._init_dirs()
|
|
assert func().split(os.sep)[-elems:] == expected
|
|
|
|
|
|
DirArgTest = collections.namedtuple('DirArgTest', 'arg, expected')
|
|
|
|
|
|
class TestArguments:
|
|
|
|
"""Tests the --basedir argument."""
|
|
|
|
@pytest.mark.parametrize('typ, args', [
|
|
('config', []),
|
|
('config', [True]), # user config
|
|
('data', []),
|
|
('cache', []),
|
|
('download', []),
|
|
pytest.param('runtime', [], marks=pytest.mark.linux)])
|
|
def test_basedir(self, tmpdir, typ, args):
|
|
"""Test --basedir."""
|
|
expected = str(tmpdir / typ)
|
|
init_args = types.SimpleNamespace(basedir=str(tmpdir))
|
|
standarddir._init_dirs(init_args)
|
|
func = getattr(standarddir, typ)
|
|
assert func(*args) == expected
|
|
|
|
def test_basedir_relative(self, tmpdir):
|
|
"""Test --basedir with a relative path."""
|
|
basedir = (tmpdir / 'basedir')
|
|
basedir.ensure(dir=True)
|
|
with tmpdir.as_cwd():
|
|
args = types.SimpleNamespace(basedir='basedir')
|
|
standarddir._init_dirs(args)
|
|
assert standarddir.config() == str(basedir / 'config')
|
|
|
|
|
|
class TestInitCacheDirTag:
|
|
|
|
"""Tests for _init_cachedir_tag."""
|
|
|
|
def test_existent_cache_dir_tag(self, tmpdir, mocker, monkeypatch):
|
|
"""Test with an existent CACHEDIR.TAG."""
|
|
monkeypatch.setattr(standarddir, 'cache', lambda: str(tmpdir))
|
|
mocker.patch('builtins.open', side_effect=AssertionError)
|
|
m = mocker.patch('qutebrowser.utils.standarddir.os')
|
|
m.path.join.side_effect = os.path.join
|
|
m.path.exists.return_value = True
|
|
standarddir._init_cachedir_tag()
|
|
assert not tmpdir.listdir()
|
|
m.path.exists.assert_called_with(str(tmpdir / 'CACHEDIR.TAG'))
|
|
|
|
def test_new_cache_dir_tag(self, tmpdir, mocker, monkeypatch):
|
|
"""Test creating a new CACHEDIR.TAG."""
|
|
monkeypatch.setattr(standarddir, 'cache', lambda: str(tmpdir))
|
|
standarddir._init_cachedir_tag()
|
|
assert tmpdir.listdir() == [(tmpdir / 'CACHEDIR.TAG')]
|
|
data = (tmpdir / 'CACHEDIR.TAG').read_text('utf-8')
|
|
assert data == textwrap.dedent("""
|
|
Signature: 8a477f597d28d172789f06886806bc55
|
|
# This file is a cache directory tag created by qutebrowser.
|
|
# For information about cache directory tags, see:
|
|
# http://www.brynosaurus.com/cachedir/
|
|
""").lstrip()
|
|
|
|
def test_open_oserror(self, caplog, tmpdir, mocker, monkeypatch):
|
|
"""Test creating a new CACHEDIR.TAG."""
|
|
monkeypatch.setattr(standarddir, 'cache', lambda: str(tmpdir))
|
|
mocker.patch('builtins.open', side_effect=OSError)
|
|
with caplog.at_level(logging.ERROR, 'init'):
|
|
standarddir._init_cachedir_tag()
|
|
assert len(caplog.records) == 1
|
|
assert caplog.records[0].message == 'Failed to create CACHEDIR.TAG'
|
|
assert not tmpdir.listdir()
|
|
|
|
|
|
class TestCreatingDir:
|
|
|
|
"""Make sure inexistent directories are created properly."""
|
|
|
|
DIR_TYPES = ['config', 'data', 'cache', 'download', 'runtime']
|
|
|
|
@pytest.mark.parametrize('typ', DIR_TYPES)
|
|
def test_basedir(self, tmpdir, typ):
|
|
"""Test --basedir."""
|
|
basedir = tmpdir / 'basedir'
|
|
assert not basedir.exists()
|
|
args = types.SimpleNamespace(basedir=str(basedir))
|
|
standarddir._init_dirs(args)
|
|
|
|
func = getattr(standarddir, typ)
|
|
func()
|
|
|
|
assert basedir.exists()
|
|
|
|
if os.name == 'posix':
|
|
assert basedir.stat().mode & 0o777 == 0o700
|
|
|
|
@pytest.mark.parametrize('typ', DIR_TYPES)
|
|
def test_exists_race_condition(self, mocker, tmpdir, typ):
|
|
"""Make sure there can't be a TOCTOU issue when creating the file.
|
|
|
|
See https://github.com/qutebrowser/qutebrowser/issues/942.
|
|
"""
|
|
(tmpdir / typ).ensure(dir=True)
|
|
|
|
m = mocker.patch('qutebrowser.utils.standarddir.os')
|
|
m.makedirs = os.makedirs
|
|
m.sep = os.sep
|
|
m.path.join = os.path.join
|
|
m.path.exists.return_value = False
|
|
m.path.abspath = lambda x: x
|
|
|
|
args = types.SimpleNamespace(basedir=str(tmpdir))
|
|
standarddir._init_dirs(args)
|
|
|
|
func = getattr(standarddir, typ)
|
|
func()
|
|
|
|
|
|
class TestSystemData:
|
|
|
|
"""Test system data path."""
|
|
|
|
def test_system_datadir_exist_linux(self, monkeypatch):
|
|
"""Test that /usr/share/qutebrowser is used if path exists."""
|
|
monkeypatch.setattr('sys.platform', "linux")
|
|
monkeypatch.setattr(os.path, 'exists', lambda path: True)
|
|
standarddir._init_dirs()
|
|
assert standarddir.data(system=True) == "/usr/share/qutebrowser"
|
|
|
|
@pytest.mark.linux
|
|
def test_system_datadir_not_exist_linux(self, monkeypatch, tmpdir,
|
|
fake_args):
|
|
"""Test that system-wide path isn't used on linux if path not exist."""
|
|
fake_args.basedir = str(tmpdir)
|
|
monkeypatch.setattr(os.path, 'exists', lambda path: False)
|
|
standarddir._init_dirs(fake_args)
|
|
assert standarddir.data(system=True) == standarddir.data()
|
|
|
|
def test_system_datadir_unsupportedos(self, monkeypatch, tmpdir,
|
|
fake_args):
|
|
"""Test that system-wide path is not used on non-Linux OS."""
|
|
fake_args.basedir = str(tmpdir)
|
|
monkeypatch.setattr('sys.platform', "potato")
|
|
standarddir._init_dirs(fake_args)
|
|
assert standarddir.data(system=True) == standarddir.data()
|
|
|
|
|
|
# FIXME:conf
|
|
# needs AppDataLocation
|
|
@pytest.mark.qt55
|
|
class TestDataMigrations:
|
|
|
|
"""Test moving various data from an old to a new location."""
|
|
|
|
@pytest.fixture(autouse=True)
|
|
def patch_standardpaths(self, files, tmpdir, monkeypatch):
|
|
locations = {
|
|
QStandardPaths.DataLocation: str(files.local_data_dir),
|
|
QStandardPaths.CacheLocation: str(tmpdir / 'cache'),
|
|
QStandardPaths.AppDataLocation: str(files.roaming_data_dir),
|
|
}
|
|
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'))
|
|
monkeypatch.setattr(
|
|
standarddir, 'config', lambda auto=False:
|
|
str(files.auto_config_dir if auto else files.config_dir))
|
|
|
|
@pytest.fixture
|
|
def files(self, tmpdir):
|
|
files = collections.namedtuple('Files', [
|
|
'old_webengine_data', 'new_webengine_data',
|
|
'old_webengine_cache', 'new_webengine_cache',
|
|
'auto_config_dir', 'config_dir',
|
|
'local_data_dir', 'roaming_data_dir'])
|
|
return files(
|
|
old_webengine_data=(tmpdir / 'data' / 'QtWebEngine' / 'Default' /
|
|
'datafile'),
|
|
new_webengine_data=tmpdir / 'new_data' / 'webengine' / 'datafile',
|
|
old_webengine_cache=(tmpdir / 'cache' / 'QtWebEngine' / 'Default' /
|
|
'cachefile'),
|
|
new_webengine_cache=(tmpdir / 'new_cache' / 'webengine' /
|
|
'cachefile'),
|
|
auto_config_dir=tmpdir / 'auto_config',
|
|
config_dir=tmpdir / 'config',
|
|
local_data_dir=tmpdir / 'data',
|
|
roaming_data_dir=tmpdir / 'roaming-data',
|
|
)
|
|
|
|
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_webengine_data.ensure()
|
|
files.old_webengine_cache.ensure()
|
|
|
|
standarddir._move_webengine_data()
|
|
|
|
assert not files.old_webengine_data.exists()
|
|
assert not files.old_webengine_cache.exists()
|
|
assert files.new_webengine_data.exists()
|
|
assert files.new_webengine_cache.exists()
|
|
|
|
@pytest.mark.parametrize('what', ['data', 'cache'])
|
|
def test_already_existing(self, files, caplog, what):
|
|
files.old_webengine_data.ensure()
|
|
files.old_webengine_cache.ensure()
|
|
|
|
if what == 'data':
|
|
files.new_webengine_data.ensure()
|
|
old_path = str(files.old_webengine_data.dirname)
|
|
new_path = str(files.new_webengine_data.dirname)
|
|
else:
|
|
files.new_webengine_cache.ensure()
|
|
old_path = str(files.old_webengine_cache.dirname)
|
|
new_path = str(files.new_webengine_cache.dirname)
|
|
|
|
with caplog.at_level(logging.ERROR):
|
|
standarddir._move_webengine_data()
|
|
|
|
record = caplog.records[-1]
|
|
expected = "Failed to move data from {} as {} is non-empty!".format(
|
|
old_path, new_path)
|
|
assert record.message == 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_webengine_data.ensure()
|
|
files.old_webengine_cache.ensure()
|
|
|
|
with caplog.at_level(logging.ERROR):
|
|
standarddir._move_webengine_data()
|
|
|
|
record = caplog.records[-1]
|
|
expected = "Failed to move data from {} to {}: error".format(
|
|
files.old_webengine_data.dirname, files.new_webengine_data.dirname)
|
|
assert record.message == expected
|
|
|
|
def test_move_macos(self, files):
|
|
"""Test moving configs on macOS."""
|
|
(files.auto_config_dir / 'autoconfig.yml').ensure()
|
|
(files.auto_config_dir / 'quickmarks').ensure()
|
|
files.config_dir.ensure(dir=True)
|
|
|
|
standarddir._move_macos()
|
|
|
|
assert (files.auto_config_dir / 'autoconfig.yml').exists()
|
|
assert not (files.config_dir / 'autoconfig.yml').exists()
|
|
assert not (files.auto_config_dir / 'quickmarks').exists()
|
|
assert (files.config_dir / 'quickmarks').exists()
|
|
|
|
def test_move_windows(self, files):
|
|
"""Test moving configs on Windows."""
|
|
(files.local_data_dir / 'data' / 'blocked-hosts').ensure()
|
|
(files.local_data_dir / 'qutebrowser.conf').ensure()
|
|
|
|
standarddir._move_windows()
|
|
|
|
assert (files.roaming_data_dir / 'data' / 'blocked-hosts').exists()
|
|
assert (files.roaming_data_dir / 'config' /
|
|
'qutebrowser.conf').exists()
|
|
|
|
|
|
@pytest.mark.parametrize('with_args', [True, False])
|
|
def test_init(mocker, tmpdir, with_args):
|
|
"""Do some sanity checks for standarddir.init().
|
|
|
|
Things like _init_cachedir_tag() and _move_webengine_data() are tested in
|
|
more detail in other tests.
|
|
"""
|
|
assert standarddir._locations == {}
|
|
|
|
if with_args:
|
|
args = types.SimpleNamespace(basedir=str(tmpdir))
|
|
m = mocker.patch('qutebrowser.utils.standarddir._move_webengine_data')
|
|
else:
|
|
args = None
|
|
|
|
standarddir.init(args)
|
|
|
|
assert standarddir._locations != {}
|
|
if with_args:
|
|
assert m.called
|
|
|
|
|
|
@pytest.mark.linux
|
|
def test_downloads_dir_not_crated(monkeypatch, tmpdir):
|
|
"""Make sure ~/Downloads is not created."""
|
|
download_dir = tmpdir / 'Downloads'
|
|
monkeypatch.setenv('HOME', str(tmpdir))
|
|
# Make sure xdg-user-dirs.dirs is not picked up
|
|
monkeypatch.delenv('XDG_CONFIG_HOME', raising=False)
|
|
standarddir._init_dirs()
|
|
assert standarddir.download() == str(download_dir)
|
|
assert not download_dir.exists()
|