2014-05-05 12:16:12 +02:00
|
|
|
# Copyright 2014 Florian Bruhin (The Compiler) <mail@qutebrowser.org>
|
|
|
|
#
|
|
|
|
# This file is part of qutebrowser.
|
|
|
|
#
|
|
|
|
# qutebrowser is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# qutebrowser is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with qutebrowser. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
2014-05-05 15:24:04 +02:00
|
|
|
# pylint: disable=missing-docstring,blacklisted-name,protected-access
|
2014-05-05 12:16:12 +02:00
|
|
|
|
|
|
|
"""Tests for qutebrowser.utils.misc."""
|
|
|
|
|
2014-05-09 08:43:31 +02:00
|
|
|
import os
|
2014-05-05 12:16:12 +02:00
|
|
|
import sys
|
2014-05-09 08:43:31 +02:00
|
|
|
import shutil
|
2014-06-10 11:57:42 +02:00
|
|
|
import argparse
|
2014-05-05 12:16:12 +02:00
|
|
|
import unittest
|
|
|
|
import os.path
|
2014-05-05 14:17:56 +02:00
|
|
|
import subprocess
|
2014-05-09 08:43:31 +02:00
|
|
|
from tempfile import mkdtemp
|
2014-05-05 12:16:12 +02:00
|
|
|
from unittest import TestCase
|
|
|
|
|
2014-05-09 08:43:31 +02:00
|
|
|
from PyQt5.QtCore import QStandardPaths, QCoreApplication
|
2014-06-12 21:43:30 +02:00
|
|
|
from PyQt5.QtGui import QColor
|
2014-05-09 08:43:31 +02:00
|
|
|
|
2014-05-05 12:16:12 +02:00
|
|
|
import qutebrowser.utils.misc as utils
|
2014-05-09 08:43:31 +02:00
|
|
|
from qutebrowser.test.helpers import environ_set_temp
|
2014-05-05 12:16:12 +02:00
|
|
|
|
|
|
|
|
2014-05-16 07:46:56 +02:00
|
|
|
class ElidingTests(TestCase):
|
|
|
|
|
|
|
|
"""Test elide."""
|
|
|
|
|
|
|
|
ELLIPSIS = '\u2026'
|
|
|
|
|
|
|
|
def test_too_small(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test eliding to 0 chars which should fail."""
|
2014-05-16 07:46:56 +02:00
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
utils.elide('foo', 0)
|
|
|
|
|
|
|
|
def test_length_one(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test eliding to 1 char which should yield ..."""
|
2014-05-16 07:46:56 +02:00
|
|
|
self.assertEqual(utils.elide('foo', 1), self.ELLIPSIS)
|
|
|
|
|
|
|
|
def test_fits(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test eliding with a string which fits exactly."""
|
2014-05-16 07:46:56 +02:00
|
|
|
self.assertEqual(utils.elide('foo', 3), 'foo')
|
|
|
|
|
|
|
|
def test_elided(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test eliding with a string which should get elided."""
|
2014-05-16 07:46:56 +02:00
|
|
|
self.assertEqual(utils.elide('foobar', 3), 'fo' + self.ELLIPSIS)
|
|
|
|
|
|
|
|
|
2014-05-14 23:29:18 +02:00
|
|
|
class CheckOverflowTests(TestCase):
|
|
|
|
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test check_overflow.
|
|
|
|
|
|
|
|
Class attributes:
|
|
|
|
INT32_MIN: Minimum valid value for a signed int32.
|
|
|
|
INT32_MAX: Maximum valid value for a signed int32.
|
|
|
|
INT64_MIN: Minimum valid value for a signed int64.
|
|
|
|
INT64_MAX: Maximum valid value for a signed int64.
|
|
|
|
GOOD_VALUES: A dict of types mapped to a list of good values.
|
|
|
|
BAD_VALUES: A dict of types mapped to a list of bad values.
|
|
|
|
"""
|
2014-05-14 23:29:18 +02:00
|
|
|
|
|
|
|
INT32_MIN = -(2 ** 31)
|
|
|
|
INT32_MAX = 2 ** 31 - 1
|
|
|
|
INT64_MIN = -(2 ** 63)
|
|
|
|
INT64_MAX = 2 ** 63 - 1
|
|
|
|
|
|
|
|
GOOD_VALUES = {
|
|
|
|
'int': [-1, 0, 1, 23.42, INT32_MIN, INT32_MAX],
|
|
|
|
'int64': [-1, 0, 1, 23.42, INT64_MIN, INT64_MAX],
|
|
|
|
}
|
|
|
|
|
|
|
|
BAD_VALUES = {
|
|
|
|
'int': [(INT32_MIN - 1, INT32_MIN),
|
|
|
|
(INT32_MAX + 1, INT32_MAX),
|
|
|
|
(float(INT32_MAX + 1), INT32_MAX)],
|
|
|
|
'int64': [(INT64_MIN - 1, INT64_MIN),
|
|
|
|
(INT64_MAX + 1, INT64_MAX),
|
|
|
|
(float(INT64_MAX + 1), INT64_MAX)],
|
|
|
|
}
|
|
|
|
|
|
|
|
def test_good_values(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test values which are inside bounds."""
|
2014-05-14 23:29:18 +02:00
|
|
|
for ctype, vals in self.GOOD_VALUES.items():
|
|
|
|
for val in vals:
|
|
|
|
utils.check_overflow(val, ctype)
|
|
|
|
|
|
|
|
def test_bad_values_fatal(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test values which are outside bounds with fatal=True."""
|
2014-05-14 23:29:18 +02:00
|
|
|
for ctype, vals in self.BAD_VALUES.items():
|
|
|
|
for (val, _) in vals:
|
|
|
|
with self.assertRaises(OverflowError, msg=ctype):
|
|
|
|
utils.check_overflow(val, ctype)
|
|
|
|
|
|
|
|
def test_bad_values_nonfatal(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test values which are outside bounds with fatal=False."""
|
2014-05-14 23:29:18 +02:00
|
|
|
for ctype, vals in self.BAD_VALUES.items():
|
|
|
|
for (val, replacement) in vals:
|
|
|
|
newval = utils.check_overflow(val, ctype, fatal=False)
|
|
|
|
self.assertEqual(newval, replacement,
|
|
|
|
"{}: {}".format(ctype, val))
|
|
|
|
|
|
|
|
|
2014-05-05 12:16:12 +02:00
|
|
|
class ReadFileTests(TestCase):
|
|
|
|
|
|
|
|
"""Test read_file."""
|
|
|
|
|
|
|
|
def test_readfile(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Read a testfile."""
|
2014-05-05 12:16:12 +02:00
|
|
|
content = utils.read_file(os.path.join('test', 'testfile'))
|
|
|
|
self.assertEqual(content.splitlines()[0], "Hello World!")
|
|
|
|
|
|
|
|
|
|
|
|
class DottedGetattrTests(TestCase):
|
|
|
|
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test dotted_getattr.
|
|
|
|
|
|
|
|
Attributes:
|
|
|
|
test: Test class instance for getattr.
|
|
|
|
"""
|
2014-05-05 12:16:12 +02:00
|
|
|
|
|
|
|
class Test:
|
|
|
|
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Sample class used to test dotted_getattr."""
|
|
|
|
|
2014-05-05 12:16:12 +02:00
|
|
|
foo = None
|
|
|
|
|
|
|
|
class Test2:
|
|
|
|
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Sample class used to test dotted_getattr."""
|
|
|
|
|
2014-05-05 12:16:12 +02:00
|
|
|
bar = 1
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
self.test = self.Test()
|
|
|
|
self.test.foo = self.Test2()
|
|
|
|
|
|
|
|
def test_dotted_getattr(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test dotted_getattr with a valid path."""
|
2014-05-05 12:16:12 +02:00
|
|
|
attr = utils.dotted_getattr(self, 'test.foo.bar')
|
|
|
|
self.assertEqual(attr, 1)
|
|
|
|
|
|
|
|
def test_invalid_path(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test dotted_getattr with an invalid path."""
|
2014-05-05 12:16:12 +02:00
|
|
|
with self.assertRaises(AttributeError):
|
|
|
|
_ = utils.dotted_getattr(self, 'test.foo.baz')
|
|
|
|
|
|
|
|
|
|
|
|
class SafeShlexSplitTests(TestCase):
|
|
|
|
|
|
|
|
"""Test safe_shlex_split."""
|
|
|
|
|
|
|
|
def test_normal(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test safe_shlex_split with a simple string."""
|
2014-05-05 12:16:12 +02:00
|
|
|
items = utils.safe_shlex_split('one two')
|
|
|
|
self.assertEqual(items, ['one', 'two'])
|
|
|
|
|
|
|
|
def test_quoted(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test safe_shlex_split with a normally quoted string."""
|
2014-05-05 12:16:12 +02:00
|
|
|
items = utils.safe_shlex_split('one "two three" four')
|
|
|
|
self.assertEqual(items, ['one', 'two three', 'four'])
|
|
|
|
|
|
|
|
def test_escaped(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test safe_shlex_split with a normal escaped string."""
|
2014-05-05 12:16:12 +02:00
|
|
|
items = utils.safe_shlex_split(r'one "two\" three" four')
|
|
|
|
self.assertEqual(items, ['one', 'two" three', 'four'])
|
|
|
|
|
|
|
|
def test_unbalanced_quotes(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test safe_shlex_split with unbalanded quotes."""
|
2014-05-05 12:16:12 +02:00
|
|
|
items = utils.safe_shlex_split(r'one "two three')
|
|
|
|
self.assertEqual(items, ['one', 'two three'])
|
|
|
|
|
|
|
|
def test_unfinished_escape(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test safe_shlex_split with an unfinished escape."""
|
2014-05-05 12:16:12 +02:00
|
|
|
items = utils.safe_shlex_split('one\\')
|
|
|
|
self.assertEqual(items, ['one\\'])
|
|
|
|
|
|
|
|
def test_both(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test safe_shlex_split with an unfinished escape and quotes.."""
|
2014-05-05 12:16:12 +02:00
|
|
|
items = utils.safe_shlex_split('one "two\\')
|
|
|
|
self.assertEqual(items, ['one', 'two\\'])
|
|
|
|
|
|
|
|
|
|
|
|
class ShellEscapeTests(TestCase):
|
|
|
|
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Tests for shell_escape.
|
|
|
|
|
|
|
|
Class attributes:
|
|
|
|
TEXTS_LINUX: A list of (input, output) of expected texts for Linux.
|
|
|
|
TEXTS_WINDOWS: A list of (input, output) of expected texts for Windows.
|
|
|
|
|
|
|
|
Attributes:
|
|
|
|
platform: The saved sys.platform value.
|
|
|
|
"""
|
|
|
|
|
2014-06-06 17:12:54 +02:00
|
|
|
TEXTS_LINUX = (
|
2014-05-05 14:17:56 +02:00
|
|
|
('', "''"),
|
|
|
|
('foo%bar+baz', 'foo%bar+baz'),
|
|
|
|
('foo$bar', "'foo$bar'"),
|
|
|
|
("$'b", """'$'"'"'b'"""),
|
2014-06-06 17:12:54 +02:00
|
|
|
)
|
2014-05-05 12:16:12 +02:00
|
|
|
|
2014-06-06 17:12:54 +02:00
|
|
|
TEXTS_WINDOWS = (
|
2014-05-05 14:17:56 +02:00
|
|
|
('', '""'),
|
|
|
|
('foo*bar?baz', 'foo*bar?baz'),
|
|
|
|
("a&b|c^d<e>f%", "a^&b^|c^^d^<e^>f^%"),
|
|
|
|
('foo"bar', 'foo"""bar'),
|
2014-06-06 17:12:54 +02:00
|
|
|
)
|
2014-05-05 12:16:12 +02:00
|
|
|
|
2014-05-05 14:17:56 +02:00
|
|
|
def setUp(self):
|
|
|
|
self.platform = sys.platform
|
2014-05-05 12:16:12 +02:00
|
|
|
|
2014-05-05 14:17:56 +02:00
|
|
|
def test_fake_linux(self):
|
|
|
|
"""Fake test which simply checks if the escaped string looks right."""
|
2014-05-05 12:16:12 +02:00
|
|
|
sys.platform = 'linux'
|
2014-05-05 14:17:56 +02:00
|
|
|
for (orig, escaped) in self.TEXTS_LINUX:
|
|
|
|
self.assertEqual(utils.shell_escape(orig), escaped)
|
2014-05-05 12:16:12 +02:00
|
|
|
|
2014-05-05 14:17:56 +02:00
|
|
|
def test_fake_windows(self):
|
|
|
|
"""Fake test which simply checks if the escaped string looks right."""
|
2014-05-05 12:16:12 +02:00
|
|
|
sys.platform = 'win32'
|
2014-05-05 14:17:56 +02:00
|
|
|
for (orig, escaped) in self.TEXTS_WINDOWS:
|
|
|
|
self.assertEqual(utils.shell_escape(orig), escaped)
|
|
|
|
|
|
|
|
@unittest.skipUnless(sys.platform.startswith("linux"), "requires Linux")
|
|
|
|
def test_real_linux(self):
|
|
|
|
"""Real test which prints an escaped string via python."""
|
|
|
|
for (orig, _escaped) in self.TEXTS_LINUX:
|
|
|
|
cmd = ("python -c 'import sys; print(sys.argv[1], end=\"\")' "
|
|
|
|
"{}".format(utils.shell_escape(orig)))
|
|
|
|
out = subprocess.check_output(cmd, shell=True).decode('ASCII')
|
|
|
|
self.assertEqual(out, orig, cmd)
|
|
|
|
|
|
|
|
@unittest.skipUnless(sys.platform.startswith("win"), "requires Windows")
|
|
|
|
def test_real_windows(self):
|
|
|
|
"""Real test which prints an escaped string via python."""
|
|
|
|
for (orig, _escaped) in self.TEXTS_WINDOWS:
|
|
|
|
cmd = ('python -c "import sys; print(sys.argv[1], end=\'\')" '
|
|
|
|
'{}'.format(utils.shell_escape(orig)))
|
|
|
|
out = subprocess.check_output(cmd, shell=True).decode('ASCII')
|
|
|
|
self.assertEqual(out, orig, cmd)
|
2014-05-05 12:16:12 +02:00
|
|
|
|
|
|
|
def tearDown(self):
|
|
|
|
sys.platform = self.platform
|
|
|
|
|
2014-05-05 15:24:04 +02:00
|
|
|
|
2014-05-09 08:43:31 +02:00
|
|
|
class GetStandardDirLinuxTests(TestCase):
|
|
|
|
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Tests for get_standard_dir under Linux.
|
|
|
|
|
|
|
|
Attributes:
|
|
|
|
temp_dir: A temporary directory.
|
|
|
|
app: The QCoreApplication used.
|
|
|
|
"""
|
|
|
|
|
2014-05-09 08:43:31 +02:00
|
|
|
def setUp(self):
|
|
|
|
self.temp_dir = mkdtemp()
|
|
|
|
self.app = QCoreApplication([])
|
|
|
|
self.app.setApplicationName('qutebrowser')
|
|
|
|
|
|
|
|
@unittest.skipUnless(sys.platform.startswith("linux"), "requires Linux")
|
|
|
|
def test_data_explicit(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test data dir with XDG_DATA_HOME explicitely set."""
|
2014-05-09 08:43:31 +02:00
|
|
|
with environ_set_temp('XDG_DATA_HOME', self.temp_dir):
|
2014-05-09 09:15:00 +02:00
|
|
|
cur_dir = utils.get_standard_dir(QStandardPaths.DataLocation)
|
|
|
|
self.assertEqual(cur_dir, os.path.join(self.temp_dir,
|
|
|
|
'qutebrowser'))
|
|
|
|
self.assertTrue(os.path.exists(cur_dir))
|
2014-05-09 08:43:31 +02:00
|
|
|
|
|
|
|
@unittest.skipUnless(sys.platform.startswith("linux"), "requires Linux")
|
|
|
|
def test_config_explicit(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test config dir with XDG_CONFIG_HOME explicitely set."""
|
2014-05-09 08:43:31 +02:00
|
|
|
with environ_set_temp('XDG_CONFIG_HOME', self.temp_dir):
|
2014-05-09 09:15:00 +02:00
|
|
|
cur_dir = utils.get_standard_dir(QStandardPaths.ConfigLocation)
|
|
|
|
self.assertEqual(cur_dir, os.path.join(self.temp_dir,
|
|
|
|
'qutebrowser'))
|
|
|
|
self.assertTrue(os.path.exists(cur_dir))
|
2014-05-09 08:43:31 +02:00
|
|
|
|
|
|
|
@unittest.skipUnless(sys.platform.startswith("linux"), "requires Linux")
|
|
|
|
def test_cache_explicit(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test cache dir with XDG_CACHE_HOME explicitely set."""
|
2014-05-09 08:43:31 +02:00
|
|
|
with environ_set_temp('XDG_CACHE_HOME', self.temp_dir):
|
2014-05-09 09:15:00 +02:00
|
|
|
cur_dir = utils.get_standard_dir(QStandardPaths.CacheLocation)
|
|
|
|
self.assertEqual(cur_dir, os.path.join(self.temp_dir,
|
|
|
|
'qutebrowser'))
|
|
|
|
self.assertTrue(os.path.exists(cur_dir))
|
2014-05-09 08:43:31 +02:00
|
|
|
|
|
|
|
@unittest.skipUnless(sys.platform.startswith("linux"), "requires Linux")
|
|
|
|
def test_data(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test data dir with XDG_DATA_HOME not set."""
|
2014-05-09 08:43:31 +02:00
|
|
|
with environ_set_temp('HOME', self.temp_dir):
|
2014-05-09 09:15:00 +02:00
|
|
|
cur_dir = utils.get_standard_dir(QStandardPaths.DataLocation)
|
|
|
|
self.assertEqual(cur_dir, os.path.join(self.temp_dir, '.local',
|
|
|
|
'share', 'qutebrowser'))
|
|
|
|
self.assertTrue(os.path.exists(cur_dir))
|
2014-05-09 08:43:31 +02:00
|
|
|
|
|
|
|
@unittest.skipUnless(sys.platform.startswith("linux"), "requires Linux")
|
|
|
|
def test_config(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test config dir with XDG_CONFIG_HOME not set."""
|
2014-05-09 08:43:31 +02:00
|
|
|
with environ_set_temp('HOME', self.temp_dir):
|
2014-05-09 09:15:00 +02:00
|
|
|
cur_dir = utils.get_standard_dir(
|
2014-05-09 08:43:31 +02:00
|
|
|
QStandardPaths.ConfigLocation)
|
2014-05-09 09:15:00 +02:00
|
|
|
self.assertEqual(cur_dir, os.path.join(self.temp_dir, '.config',
|
|
|
|
'qutebrowser'))
|
|
|
|
self.assertTrue(os.path.exists(cur_dir))
|
2014-05-09 08:43:31 +02:00
|
|
|
|
|
|
|
@unittest.skipUnless(sys.platform.startswith("linux"), "requires Linux")
|
|
|
|
def test_cache(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test cache dir with XDG_CACHE_HOME not set."""
|
2014-05-09 08:43:31 +02:00
|
|
|
with environ_set_temp('HOME', self.temp_dir):
|
2014-05-09 09:15:00 +02:00
|
|
|
cur_dir = utils.get_standard_dir(QStandardPaths.CacheLocation)
|
|
|
|
self.assertEqual(cur_dir, os.path.join(self.temp_dir, '.cache',
|
2014-05-09 09:20:11 +02:00
|
|
|
'qutebrowser'))
|
2014-05-09 09:15:00 +02:00
|
|
|
self.assertTrue(os.path.exists(cur_dir))
|
2014-05-09 08:43:31 +02:00
|
|
|
|
|
|
|
def tearDown(self):
|
|
|
|
self.app.quit()
|
|
|
|
shutil.rmtree(self.temp_dir)
|
|
|
|
|
|
|
|
|
2014-05-09 09:16:24 +02:00
|
|
|
class GetStandardDirWindowsTests(TestCase):
|
|
|
|
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Tests for get_standard_dir under Windows.
|
|
|
|
|
|
|
|
Attributes:
|
|
|
|
app: The QCoreApplication used.
|
|
|
|
"""
|
|
|
|
|
2014-05-09 09:16:24 +02:00
|
|
|
def setUp(self):
|
|
|
|
self.app = QCoreApplication([])
|
|
|
|
# We can't store the files in a temp dir, so we don't chose qutebrowser
|
|
|
|
self.app.setApplicationName('qutebrowser_test')
|
|
|
|
|
|
|
|
@unittest.skipUnless(sys.platform.startswith("win"), "requires Windows")
|
|
|
|
def test_data(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test data dir."""
|
2014-05-09 09:16:24 +02:00
|
|
|
cur_dir = utils.get_standard_dir(QStandardPaths.DataLocation)
|
|
|
|
self.assertEqual(cur_dir.split(os.sep)[-1], 'qutebrowser_test',
|
|
|
|
cur_dir)
|
|
|
|
self.assertTrue(os.path.exists(cur_dir))
|
|
|
|
# We clean up here as we don't dare to clean up if the path doesn't end
|
|
|
|
# with qutebrowser_test - it could be *anywhere* after all.
|
|
|
|
shutil.rmtree(cur_dir)
|
|
|
|
|
|
|
|
@unittest.skipUnless(sys.platform.startswith("win"), "requires Windows")
|
|
|
|
def test_config(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test config dir."""
|
2014-05-09 09:16:24 +02:00
|
|
|
cur_dir = utils.get_standard_dir(QStandardPaths.ConfigLocation)
|
|
|
|
self.assertEqual(cur_dir.split(os.sep)[-1], 'qutebrowser_test',
|
|
|
|
cur_dir)
|
|
|
|
self.assertTrue(os.path.exists(cur_dir))
|
|
|
|
# We clean up here as we don't dare to clean up if the path doesn't end
|
|
|
|
# with qutebrowser_test - it could be *anywhere* after all.
|
|
|
|
shutil.rmtree(cur_dir)
|
|
|
|
|
|
|
|
@unittest.skipUnless(sys.platform.startswith("win"), "requires Windows")
|
|
|
|
def test_cache(self):
|
2014-05-27 13:06:13 +02:00
|
|
|
"""Test cache dir."""
|
2014-05-09 09:16:24 +02:00
|
|
|
cur_dir = utils.get_standard_dir(QStandardPaths.CacheLocation)
|
|
|
|
self.assertEqual(cur_dir.split(os.sep)[-2:],
|
|
|
|
['qutebrowser_test', 'cache'], cur_dir)
|
|
|
|
self.assertTrue(os.path.exists(cur_dir))
|
|
|
|
# We clean up here as we don't dare to clean up if the path doesn't end
|
|
|
|
# with qutebrowser_test - it could be *anywhere* after all.
|
|
|
|
shutil.rmtree(cur_dir)
|
|
|
|
|
|
|
|
def tearDown(self):
|
|
|
|
self.app.quit()
|
|
|
|
|
|
|
|
|
2014-06-10 11:57:42 +02:00
|
|
|
class GetQtArgsTests(TestCase):
|
|
|
|
|
|
|
|
"""Tests for get_qt_args."""
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
self.parser = argparse.ArgumentParser()
|
|
|
|
|
|
|
|
def _namespace(self, cmdline, flags=None, args=None):
|
|
|
|
"""Get an argparse namespace object based on arguments given.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
cmdline: The given commandline.
|
|
|
|
flags: A list of strings (argument names) for flags without an
|
|
|
|
argument.
|
|
|
|
args: A list of arguemnt names for flags with an argument.
|
|
|
|
"""
|
|
|
|
if flags is not None:
|
|
|
|
for e in flags:
|
|
|
|
self.parser.add_argument(e, action='store_true')
|
|
|
|
if args is not None:
|
|
|
|
for e in args:
|
|
|
|
self.parser.add_argument(e, nargs=1)
|
|
|
|
return self.parser.parse_args(cmdline)
|
|
|
|
|
|
|
|
def test_no_qt_args(self):
|
|
|
|
"""Test commandline with no Qt arguments given."""
|
|
|
|
ns = self._namespace(['--foo'], flags=['--foo'])
|
|
|
|
self.assertEqual(utils.get_qt_args(ns), [sys.argv[0]])
|
|
|
|
|
|
|
|
def test_qt_flag(self):
|
|
|
|
"""Test commandline with a Qt flag."""
|
|
|
|
ns = self._namespace(['--foo', '--qt-reverse', '--bar'],
|
|
|
|
flags=['--foo', '--qt-reverse', '--bar'])
|
|
|
|
self.assertEqual(utils.get_qt_args(ns), [sys.argv[0], '-reverse'])
|
|
|
|
|
|
|
|
def test_qt_arg(self):
|
|
|
|
"""Test commandline with a Qt argument."""
|
|
|
|
ns = self._namespace(['--qt-stylesheet', 'foobar'],
|
|
|
|
args=['--qt-stylesheet'])
|
|
|
|
self.assertEqual(utils.get_qt_args(ns), [sys.argv[0], '-stylesheet',
|
|
|
|
'foobar'])
|
|
|
|
|
|
|
|
def test_qt_both(self):
|
|
|
|
"""Test commandline with a Qt argument and flag."""
|
|
|
|
ns = self._namespace(['--qt-stylesheet', 'foobar', '--qt-reverse'],
|
|
|
|
flags=['--qt-reverse'], args=['--qt-stylesheet'])
|
2014-06-12 15:17:38 +02:00
|
|
|
qt_args = utils.get_qt_args(ns)
|
|
|
|
self.assertEqual(qt_args[0], sys.argv[0])
|
|
|
|
self.assertIn('-reverse', qt_args)
|
|
|
|
self.assertIn('-stylesheet', qt_args)
|
|
|
|
self.assertIn('foobar', qt_args)
|
2014-06-10 11:57:42 +02:00
|
|
|
|
|
|
|
|
2014-06-12 21:43:30 +02:00
|
|
|
class InterpolateColorTests(TestCase):
|
|
|
|
|
|
|
|
"""Tests for interpolate_color.
|
|
|
|
|
|
|
|
Attributes:
|
|
|
|
white: The QColor white as a valid QColor for tests.
|
|
|
|
white: The QColor black as a valid QColor for tests.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
self.white = QColor('white')
|
|
|
|
self.black = QColor('black')
|
|
|
|
|
|
|
|
def test_invalid_start(self):
|
|
|
|
"""Test an invalid start color."""
|
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
utils.interpolate_color(QColor(), self.white, 0)
|
|
|
|
|
|
|
|
def test_invalid_end(self):
|
|
|
|
"""Test an invalid end color."""
|
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
utils.interpolate_color(self.white, QColor(), 0)
|
|
|
|
|
|
|
|
def test_invalid_percentage(self):
|
|
|
|
"""Test an invalid percentage."""
|
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
utils.interpolate_color(self.white, self.white, -1)
|
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
utils.interpolate_color(self.white, self.white, 101)
|
|
|
|
|
|
|
|
def test_invalid_colorspace(self):
|
|
|
|
"""Test an invalid colorspace."""
|
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
utils.interpolate_color(self.white, self.black, 10, QColor.Cmyk)
|
|
|
|
|
|
|
|
def test_valid_percentages_rgb(self):
|
|
|
|
"""Test 0% and 100% in the RGB colorspace."""
|
|
|
|
white = utils.interpolate_color(self.white, self.black, 0, QColor.Rgb)
|
|
|
|
black = utils.interpolate_color(self.white, self.black, 100,
|
|
|
|
QColor.Rgb)
|
|
|
|
self.assertEqual(white, self.white)
|
|
|
|
self.assertEqual(black, self.black)
|
|
|
|
|
|
|
|
def test_valid_percentages_hsv(self):
|
|
|
|
"""Test 0% and 100% in the HSV colorspace."""
|
|
|
|
white = utils.interpolate_color(self.white, self.black, 0, QColor.Hsv)
|
|
|
|
black = utils.interpolate_color(self.white, self.black, 100,
|
|
|
|
QColor.Hsv)
|
|
|
|
self.assertEqual(white, self.white)
|
|
|
|
self.assertEqual(black, self.black)
|
|
|
|
|
|
|
|
def test_valid_percentages_hsl(self):
|
|
|
|
"""Test 0% and 100% in the HSL colorspace."""
|
|
|
|
white = utils.interpolate_color(self.white, self.black, 0, QColor.Hsl)
|
|
|
|
black = utils.interpolate_color(self.white, self.black, 100,
|
|
|
|
QColor.Hsl)
|
|
|
|
self.assertEqual(white, self.white)
|
|
|
|
self.assertEqual(black, self.black)
|
|
|
|
|
|
|
|
def test_interpolation_rgb(self):
|
|
|
|
"""Test an interpolation in the RGB colorspace."""
|
|
|
|
color = utils.interpolate_color(QColor(0, 40, 100), QColor(0, 20, 200),
|
|
|
|
50, QColor.Rgb)
|
|
|
|
self.assertEqual(color, QColor(0, 30, 150))
|
|
|
|
|
|
|
|
def test_interpolation_hsv(self):
|
|
|
|
"""Test an interpolation in the HSV colorspace."""
|
|
|
|
start = QColor()
|
|
|
|
stop = QColor()
|
|
|
|
start.setHsv(0, 40, 100)
|
|
|
|
stop.setHsv(0, 20, 200)
|
|
|
|
color = utils.interpolate_color(start, stop, 50, QColor.Hsv)
|
|
|
|
expected = QColor()
|
|
|
|
expected.setHsv(0, 30, 150)
|
|
|
|
self.assertEqual(color, expected)
|
|
|
|
|
|
|
|
def test_interpolation_hsl(self):
|
|
|
|
"""Test an interpolation in the HSL colorspace."""
|
|
|
|
start = QColor()
|
|
|
|
stop = QColor()
|
|
|
|
start.setHsl(0, 40, 100)
|
|
|
|
stop.setHsl(0, 20, 200)
|
|
|
|
color = utils.interpolate_color(start, stop, 50, QColor.Hsl)
|
|
|
|
expected = QColor()
|
|
|
|
expected.setHsl(0, 30, 150)
|
|
|
|
self.assertEqual(color, expected)
|
|
|
|
|
|
|
|
|
2014-06-12 23:29:34 +02:00
|
|
|
class FormatSecondsTests(TestCase):
|
|
|
|
|
|
|
|
"""Tests for format_seconds.
|
|
|
|
|
|
|
|
Class attributes:
|
|
|
|
TESTS: A list of (input, output) tuples.
|
|
|
|
"""
|
|
|
|
|
|
|
|
TESTS = [
|
|
|
|
(-1, '-0:01'),
|
|
|
|
(0, '0:00'),
|
|
|
|
(59, '0:59'),
|
|
|
|
(60, '1:00'),
|
|
|
|
(60.4, '1:00'),
|
|
|
|
(61, '1:01'),
|
|
|
|
(-61, '-1:01'),
|
|
|
|
(3599, '59:59'),
|
|
|
|
(3600, '1:00:00'),
|
|
|
|
(3601, '1:00:01'),
|
|
|
|
(36000, '10:00:00'),
|
|
|
|
]
|
|
|
|
|
|
|
|
def test_format_seconds(self):
|
|
|
|
"""Test format_seconds with several tests."""
|
|
|
|
for seconds, out in self.TESTS:
|
|
|
|
self.assertEqual(utils.format_seconds(seconds), out, seconds)
|
|
|
|
|
|
|
|
|
2014-06-13 07:13:47 +02:00
|
|
|
class FormatSizeTests(TestCase):
|
|
|
|
|
|
|
|
"""Tests for format_size.
|
|
|
|
|
|
|
|
Class attributes:
|
|
|
|
TESTS: A list of (input, output) tuples.
|
|
|
|
"""
|
|
|
|
|
|
|
|
TESTS = [
|
|
|
|
(-1024, '-1.00k'),
|
|
|
|
(-1, '-1.00'),
|
|
|
|
(0, '0.00'),
|
|
|
|
(1023, '1023.00'),
|
|
|
|
(1024, '1.00k'),
|
|
|
|
(1034.24, '1.01k'),
|
|
|
|
(1024 * 1024 * 2, '2.00M'),
|
|
|
|
(1024 ** 10, '1024.00Y'),
|
|
|
|
(None, '?.??'),
|
|
|
|
]
|
|
|
|
|
|
|
|
def test_format_size(self):
|
|
|
|
"""Test format_size with several tests."""
|
|
|
|
for size, out in self.TESTS:
|
|
|
|
self.assertEqual(utils.format_size(size), out, size)
|
|
|
|
|
|
|
|
def test_suffix(self):
|
|
|
|
"""Test the suffix option."""
|
|
|
|
for size, out in self.TESTS:
|
|
|
|
self.assertEqual(utils.format_size(size, suffix='B'), out + 'B',
|
|
|
|
size)
|
|
|
|
|
|
|
|
def test_base(self):
|
|
|
|
"""Test with an alternative base."""
|
|
|
|
kilo_tests = [(999, '999.00'), (1000, '1.00k'), (1010, '1.01k')]
|
|
|
|
for size, out in kilo_tests:
|
|
|
|
self.assertEqual(utils.format_size(size, base=1000), out, size)
|
|
|
|
|
|
|
|
|
2014-05-05 12:16:12 +02:00
|
|
|
if __name__ == '__main__':
|
|
|
|
unittest.main()
|