qutebrowser/tests/unit/browser/test_qutescheme.py

202 lines
6.6 KiB
Python
Raw Normal View History

2017-02-08 09:28:04 +01:00
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
2018-02-05 12:19:50 +01:00
# Copyright 2017-2018 Imran Sobir
2017-02-08 09:28:04 +01:00
#
# 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/>.
2017-02-26 13:07:30 +01:00
import json
2017-02-27 18:37:24 +01:00
import os
2017-02-26 13:07:30 +01:00
import time
2018-09-05 22:51:28 +02:00
import logging
2017-02-09 11:44:51 +01:00
2017-02-08 09:28:04 +01:00
from PyQt5.QtCore import QUrl
2017-02-09 11:44:51 +01:00
import pytest
2018-09-05 22:51:28 +02:00
from qutebrowser.browser import qutescheme, pdfjs
2017-02-08 09:28:04 +01:00
2017-02-27 18:37:24 +01:00
class TestJavascriptHandler:
"""Test the qute://javascript endpoint."""
# Tuples of fake JS files and their content.
js_files = [
('foo.js', "var a = 'foo';"),
('bar.js', "var a = 'bar';"),
]
@pytest.fixture(autouse=True)
def patch_read_file(self, monkeypatch):
"""Patch utils.read_file to return few fake JS files."""
def _read_file(path, binary=False):
2017-02-28 15:49:00 +01:00
"""Faked utils.read_file."""
2017-02-27 18:37:24 +01:00
assert not binary
for filename, content in self.js_files:
if path == os.path.join('javascript', filename):
return content
2017-03-09 17:47:12 +01:00
raise OSError("File not found {}!".format(path))
2017-02-27 18:37:24 +01:00
monkeypatch.setattr('qutebrowser.utils.utils.read_file', _read_file)
@pytest.mark.parametrize("filename, content", js_files)
def test_qutejavascript(self, filename, content):
url = QUrl("qute://javascript/{}".format(filename))
_mimetype, data = qutescheme.qute_javascript(url)
assert data == content
def test_qutejavascript_404(self):
2017-02-27 18:37:24 +01:00
url = QUrl("qute://javascript/404.js")
2018-07-10 00:55:29 +02:00
with pytest.raises(qutescheme.SchemeOSError):
qutescheme.data_for_url(url)
2017-02-27 18:37:24 +01:00
def test_qutejavascript_empty_query(self):
url = QUrl("qute://javascript")
2018-07-10 01:17:19 +02:00
with pytest.raises(qutescheme.UrlInvalidError):
2017-02-27 18:37:24 +01:00
qutescheme.qute_javascript(url)
2017-02-08 09:28:04 +01:00
class TestHistoryHandler:
2017-02-09 11:44:51 +01:00
2017-02-08 09:28:04 +01:00
"""Test the qute://history endpoint."""
@pytest.fixture(scope="module")
def now(self):
return int(time.time())
2017-02-28 15:49:00 +01:00
@pytest.fixture
def entries(self, now):
2017-02-26 13:07:30 +01:00
"""Create fake history entries."""
# create 12 history items spaced 6 hours apart, starting from now
entry_count = 12
interval = 6 * 60 * 60
items = []
for i in range(entry_count):
entry_atime = now - i * interval
entry = {"atime": str(entry_atime),
"url": QUrl("http://www.x.com/" + str(i)),
"title": "Page " + str(i)}
2017-02-26 13:07:30 +01:00
items.insert(0, entry)
return items
2017-02-08 09:28:04 +01:00
2017-02-09 18:27:32 +01:00
@pytest.fixture(autouse=True)
2018-09-06 20:04:11 +02:00
def fake_history(self, web_history, fake_args, entries):
2017-02-26 13:07:30 +01:00
"""Create fake history."""
2017-09-17 11:49:42 +02:00
fake_args.debug_flags = []
2017-02-26 13:07:30 +01:00
for item in entries:
2018-09-06 20:04:11 +02:00
web_history.add_url(**item)
2017-02-26 13:07:30 +01:00
@pytest.mark.parametrize("start_time_offset, expected_item_count", [
2017-03-02 18:40:24 +01:00
(0, 4),
(24*60*60, 4),
(48*60*60, 4),
(72*60*60, 0)
2017-02-26 13:07:30 +01:00
])
def test_qutehistory_data(self, start_time_offset, expected_item_count,
now):
2017-02-26 13:07:30 +01:00
"""Ensure qute://history/data returns correct items."""
start_time = now - start_time_offset
2017-02-26 13:07:30 +01:00
url = QUrl("qute://history/data?start_time=" + str(start_time))
2017-02-08 09:28:04 +01:00
_mimetype, data = qutescheme.qute_history(url)
2017-02-26 13:07:30 +01:00
items = json.loads(data)
2017-02-09 13:18:57 +01:00
2017-02-26 13:07:30 +01:00
assert len(items) == expected_item_count
2017-02-09 13:18:57 +01:00
# test times
2017-02-26 13:07:30 +01:00
end_time = start_time - 24*60*60
2017-03-02 18:40:24 +01:00
for item in items:
assert item['time'] <= start_time
assert item['time'] > end_time
2017-02-09 21:56:06 +01:00
2018-09-06 20:04:11 +02:00
def test_exclude(self, web_history, now, config_stub):
"""Make sure the completion.web_history.exclude setting is not used."""
config_stub.val.completion.web_history.exclude = ['www.x.com']
url = QUrl("qute://history/data?start_time={}".format(now))
_mimetype, data = qutescheme.qute_history(url)
items = json.loads(data)
assert items
2018-09-06 20:04:11 +02:00
def test_qute_history_benchmark(self, web_history, benchmark, now):
r = range(100000)
entries = {
'atime': [int(now - t) for t in r],
'url': ['www.x.com/{}'.format(t) for t in r],
'title': ['x at {}'.format(t) for t in r],
'redirect': [False for _ in r],
}
2017-02-09 21:56:06 +01:00
2018-09-06 20:04:11 +02:00
web_history.insert_batch(entries)
url = QUrl("qute://history/data?start_time={}".format(now))
Fix qute_history benchmark. This benchmark was running very quickly due to an improper setup. The current history implementation expects that a newly inserted entry must be more recent than any existing entries and sorts according to this assumption. The benchmark test inserts increasingly older entries, breaking this invariant. When run in the benchmark, the qute://history/data implementation would see an entry older than the oldest time in the time window and would immediately return with a single "next" entry. This patch inserts data in an order that mantains history's invariant and adds a sanity-check at the end of the test. It does not check for the exact length as not all entries will be within the time window. The length will be some values <= 100000, the check just ensures that there is at least something more than a "next" entry. Before: ---------------------------------------------- benchmark: 1 tests ---------------------------------------------- Name (time in us) Min Max Mean StdDev Median IQR Outliers(*) Rounds Iterations ---------------------------------------------------------------------------------------------------------------- test_qute_history_benchmark 9.3050 21.9250 9.6143 0.2454 9.5880 0.1070 230;360 9930 1 ---------------------------------------------------------------------------------------------------------------- After: -------------------------------------------------- benchmark: 1 tests ------------------------------------------------- Name (time in ms) Min Max Mean StdDev Median IQR Outliers(*) Rounds Iterations ----------------------------------------------------------------------------------------------------------------------- test_qute_history_benchmark 220.7040 223.1900 221.7536 1.1070 221.1939 1.8803 1;0 5 1 -----------------------------------------------------------------------------------------------------------------------
2017-04-16 15:01:46 +02:00
_mimetype, data = benchmark(qutescheme.qute_history, url)
assert len(json.loads(data)) > 1
class TestHelpHandler:
"""Tests for qute://help."""
@pytest.fixture
def data_patcher(self, monkeypatch):
def _patch(path, data):
def _read_file(name, binary=False):
assert path == name
if binary:
return data
return data.decode('utf-8')
monkeypatch.setattr(qutescheme.utils, 'read_file', _read_file)
return _patch
def test_unknown_file_type(self, data_patcher):
data_patcher('html/doc/foo.bin', b'\xff')
mimetype, data = qutescheme.qute_help(QUrl('qute://help/foo.bin'))
assert mimetype == 'application/octet-stream'
assert data == b'\xff'
2018-09-05 22:51:28 +02:00
class TestPDFJSHandler:
"""Test the qute://pdfjs endpoint."""
@pytest.fixture(autouse=True)
def fake_pdfjs(self, monkeypatch):
def get_pdfjs_res(path):
if path == '/existing/file.html':
return b'foobar'
raise pdfjs.PDFJSNotFound(path)
monkeypatch.setattr(pdfjs, 'get_pdfjs_res', get_pdfjs_res)
def test_existing_resource(self):
"""Test with a resource that exists."""
_mimetype, data = qutescheme.data_for_url(
QUrl('qute://pdfjs/existing/file.html'))
assert data == b'foobar'
def test_nonexisting_resource(self, caplog):
"""Test with a resource that does not exist."""
with caplog.at_level(logging.WARNING, 'misc'):
with pytest.raises(qutescheme.NotFoundError):
qutescheme.data_for_url(QUrl('qute://pdfjs/no/file.html'))
assert len(caplog.records) == 1
assert (caplog.records[0].message ==
'pdfjs resource requested but not found: /no/file.html')