2015-09-18 20:11:58 +02:00
|
|
|
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
|
|
|
|
|
2017-05-09 21:37:03 +02:00
|
|
|
# Copyright 2015-2017 Florian Bruhin (The Compiler) <mail@qutebrowser.org>
|
2015-09-18 20:11:58 +02: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-09-19 10:35:54 +02:00
|
|
|
"""Fixtures for the server webserver."""
|
2015-09-18 20:11:58 +02:00
|
|
|
|
2016-01-12 22:48:38 +01:00
|
|
|
import re
|
2015-10-10 18:16:47 +02:00
|
|
|
import sys
|
2015-11-26 14:37:47 +01:00
|
|
|
import json
|
2015-10-10 18:16:47 +02:00
|
|
|
import os.path
|
2016-01-14 22:34:26 +01:00
|
|
|
import http.client
|
2015-09-18 20:11:58 +02:00
|
|
|
|
2017-09-19 22:18:02 +02:00
|
|
|
import attr
|
2015-09-18 20:11:58 +02:00
|
|
|
import pytest
|
2016-01-15 07:13:45 +01:00
|
|
|
from PyQt5.QtCore import pyqtSignal, QUrl
|
2015-10-10 17:20:20 +02:00
|
|
|
|
2016-05-29 18:20:00 +02:00
|
|
|
from end2end.fixtures import testprocess
|
2015-09-18 20:11:58 +02:00
|
|
|
|
2016-08-03 13:08:55 +02:00
|
|
|
from qutebrowser.utils import utils
|
|
|
|
|
2015-09-18 20:11:58 +02:00
|
|
|
|
2015-11-16 23:14:24 +01:00
|
|
|
class Request(testprocess.Line):
|
2015-09-18 20:11:58 +02:00
|
|
|
|
2017-09-19 10:35:54 +02:00
|
|
|
"""A parsed line from the flask log output.
|
2015-09-18 20:11:58 +02:00
|
|
|
|
2015-11-16 23:14:24 +01:00
|
|
|
Attributes:
|
2015-11-26 14:37:47 +01:00
|
|
|
verb/path/status: Parsed from the log output.
|
2015-09-18 20:11:58 +02:00
|
|
|
"""
|
|
|
|
|
2015-11-16 23:14:24 +01:00
|
|
|
def __init__(self, data):
|
|
|
|
super().__init__(data)
|
2015-11-26 14:37:47 +01:00
|
|
|
try:
|
|
|
|
parsed = json.loads(data)
|
|
|
|
except ValueError:
|
2015-11-16 23:14:24 +01:00
|
|
|
raise testprocess.InvalidLine(data)
|
|
|
|
|
2015-11-26 14:37:47 +01:00
|
|
|
assert isinstance(parsed, dict)
|
|
|
|
assert set(parsed.keys()) == {'path', 'verb', 'status'}
|
2015-11-16 23:14:24 +01:00
|
|
|
|
2015-11-26 14:37:47 +01:00
|
|
|
self.verb = parsed['verb']
|
2015-11-16 23:14:24 +01:00
|
|
|
|
2015-11-26 14:37:47 +01:00
|
|
|
path = parsed['path']
|
2015-11-25 10:36:27 +01:00
|
|
|
self.path = '/' if path == '/' else path.rstrip('/')
|
|
|
|
|
2015-11-26 14:37:47 +01:00
|
|
|
self.status = parsed['status']
|
2016-01-15 07:13:45 +01:00
|
|
|
self._check_status()
|
|
|
|
|
|
|
|
def _check_status(self):
|
|
|
|
"""Check if the http status is what we expected."""
|
2016-01-17 21:27:24 +01:00
|
|
|
# WORKAROUND for https://github.com/PyCQA/pylint/issues/399 (?)
|
2017-05-17 19:08:59 +02:00
|
|
|
# pylint: disable=no-member
|
2016-01-15 07:13:45 +01:00
|
|
|
path_to_statuses = {
|
|
|
|
'/favicon.ico': [http.client.NOT_FOUND],
|
|
|
|
'/does-not-exist': [http.client.NOT_FOUND],
|
2016-09-10 15:51:49 +02:00
|
|
|
'/does-not-exist-2': [http.client.NOT_FOUND],
|
2017-09-19 10:35:54 +02:00
|
|
|
'/404': [http.client.NOT_FOUND],
|
2016-11-04 07:21:04 +01:00
|
|
|
|
2017-09-19 10:35:54 +02:00
|
|
|
'/redirect-later': [http.client.FOUND],
|
|
|
|
'/redirect-self': [http.client.FOUND],
|
2016-06-08 14:42:55 +02:00
|
|
|
'/redirect-to': [http.client.FOUND],
|
2017-09-19 10:35:54 +02:00
|
|
|
'/relative-redirect': [http.client.FOUND],
|
|
|
|
'/absolute-redirect': [http.client.FOUND],
|
2016-11-04 07:21:04 +01:00
|
|
|
|
2016-06-08 16:34:42 +02:00
|
|
|
'/cookies/set': [http.client.FOUND],
|
2017-03-22 13:38:03 +01:00
|
|
|
|
2017-09-19 10:35:54 +02:00
|
|
|
'/500-inline': [http.client.INTERNAL_SERVER_ERROR],
|
2016-01-15 07:13:45 +01:00
|
|
|
}
|
2016-09-10 15:51:49 +02:00
|
|
|
for i in range(15):
|
|
|
|
path_to_statuses['/redirect/{}'.format(i)] = [http.client.FOUND]
|
2016-11-09 21:49:36 +01:00
|
|
|
for suffix in ['', '1', '2', '3', '4', '5', '6']:
|
2016-11-04 07:21:04 +01:00
|
|
|
key = '/basic-auth/user{}/password{}'.format(suffix, suffix)
|
|
|
|
path_to_statuses[key] = [http.client.UNAUTHORIZED, http.client.OK]
|
|
|
|
|
2016-08-18 17:44:35 +02:00
|
|
|
default_statuses = [http.client.OK, http.client.NOT_MODIFIED]
|
2017-10-22 22:16:02 +02:00
|
|
|
# pylint: enable=no-member
|
2016-01-15 07:13:45 +01:00
|
|
|
|
|
|
|
sanitized = QUrl('http://localhost' + self.path).path() # Remove ?foo
|
2016-08-18 17:44:35 +02:00
|
|
|
expected_statuses = path_to_statuses.get(sanitized, default_statuses)
|
2016-08-25 22:58:14 +02:00
|
|
|
if self.status not in expected_statuses:
|
|
|
|
raise AssertionError(
|
|
|
|
"{} loaded with status {} but expected {}".format(
|
|
|
|
sanitized, self.status,
|
|
|
|
' / '.join(repr(e) for e in expected_statuses)))
|
2015-11-16 23:14:24 +01:00
|
|
|
|
|
|
|
def __eq__(self, other):
|
|
|
|
return NotImplemented
|
|
|
|
|
|
|
|
|
2017-09-19 22:18:02 +02:00
|
|
|
@attr.s(frozen=True, cmp=False, hash=True)
|
2015-11-16 23:14:24 +01:00
|
|
|
class ExpectedRequest:
|
|
|
|
|
|
|
|
"""Class to compare expected requests easily."""
|
|
|
|
|
2017-09-19 22:18:02 +02:00
|
|
|
verb = attr.ib()
|
|
|
|
path = attr.ib()
|
2015-11-16 23:14:24 +01:00
|
|
|
|
2015-11-18 19:56:49 +01:00
|
|
|
@classmethod
|
|
|
|
def from_request(cls, request):
|
|
|
|
"""Create an ExpectedRequest from a Request."""
|
|
|
|
return cls(request.verb, request.path)
|
|
|
|
|
2015-11-16 23:14:24 +01:00
|
|
|
def __eq__(self, other):
|
|
|
|
if isinstance(other, (Request, ExpectedRequest)):
|
2016-04-27 18:30:54 +02:00
|
|
|
return self.verb == other.verb and self.path == other.path
|
2015-11-16 23:14:24 +01:00
|
|
|
else:
|
|
|
|
return NotImplemented
|
|
|
|
|
|
|
|
|
2016-01-12 22:48:38 +01:00
|
|
|
class WebserverProcess(testprocess.Process):
|
2015-11-16 23:14:24 +01:00
|
|
|
|
2017-09-19 10:35:54 +02:00
|
|
|
"""Abstraction over a running Flask server process.
|
2015-11-16 23:14:24 +01:00
|
|
|
|
|
|
|
Reads the log from its stdout and parses it.
|
|
|
|
|
|
|
|
Signals:
|
|
|
|
new_request: Emitted when there's a new request received.
|
|
|
|
"""
|
|
|
|
|
|
|
|
new_request = pyqtSignal(Request)
|
|
|
|
Request = Request # So it can be used from the fixture easily.
|
|
|
|
ExpectedRequest = ExpectedRequest
|
|
|
|
|
2015-11-13 23:26:14 +01:00
|
|
|
KEYS = ['verb', 'path']
|
|
|
|
|
2016-01-12 22:48:38 +01:00
|
|
|
def __init__(self, script, parent=None):
|
2015-09-18 20:11:58 +02:00
|
|
|
super().__init__(parent)
|
2016-01-12 22:48:38 +01:00
|
|
|
self._script = script
|
2016-08-03 13:08:55 +02:00
|
|
|
self.port = utils.random_port()
|
2015-10-10 17:20:20 +02:00
|
|
|
self.new_data.connect(self.new_request)
|
2015-09-18 20:11:58 +02:00
|
|
|
|
|
|
|
def get_requests(self):
|
2015-10-10 17:20:20 +02:00
|
|
|
"""Get the requests to the server during this test."""
|
2015-10-13 07:11:02 +02:00
|
|
|
requests = self._get_data()
|
|
|
|
return [r for r in requests if r.path != '/favicon.ico']
|
2015-10-10 17:20:20 +02:00
|
|
|
|
|
|
|
def _parse_line(self, line):
|
2015-12-16 20:17:29 +01:00
|
|
|
self._log(line)
|
2016-01-12 22:48:38 +01:00
|
|
|
started_re = re.compile(r' \* Running on https?://127\.0\.0\.1:{}/ '
|
|
|
|
r'\(Press CTRL\+C to quit\)'.format(self.port))
|
|
|
|
if started_re.fullmatch(line):
|
2015-10-10 17:20:20 +02:00
|
|
|
self.ready.emit()
|
|
|
|
return None
|
2015-11-16 23:14:24 +01:00
|
|
|
return Request(line)
|
2015-10-10 17:20:20 +02:00
|
|
|
|
|
|
|
def _executable_args(self):
|
2015-10-10 18:16:47 +02:00
|
|
|
if hasattr(sys, 'frozen'):
|
|
|
|
executable = os.path.join(os.path.dirname(sys.executable),
|
2016-01-12 22:48:38 +01:00
|
|
|
self._script)
|
2016-01-20 06:53:25 +01:00
|
|
|
args = []
|
2015-10-10 18:16:47 +02:00
|
|
|
else:
|
|
|
|
executable = sys.executable
|
|
|
|
py_file = os.path.join(os.path.dirname(__file__),
|
2016-01-12 22:48:38 +01:00
|
|
|
self._script + '.py')
|
2016-01-20 06:53:25 +01:00
|
|
|
args = [py_file]
|
2015-10-10 18:16:47 +02:00
|
|
|
return executable, args
|
2015-09-18 20:11:58 +02:00
|
|
|
|
2016-01-20 06:53:25 +01:00
|
|
|
def _default_args(self):
|
|
|
|
return [str(self.port)]
|
|
|
|
|
2015-09-18 20:11:58 +02:00
|
|
|
def cleanup(self):
|
|
|
|
"""Clean up and shut down the process."""
|
|
|
|
self.proc.terminate()
|
|
|
|
self.proc.waitForFinished()
|
|
|
|
|
|
|
|
|
2016-08-22 07:40:24 +02:00
|
|
|
@pytest.fixture(scope='session', autouse=True)
|
2017-09-19 10:35:54 +02:00
|
|
|
def server(qapp):
|
|
|
|
"""Fixture for an server object which ensures clean setup/teardown."""
|
|
|
|
server = WebserverProcess('webserver_sub')
|
|
|
|
server.start()
|
|
|
|
yield server
|
|
|
|
server.cleanup()
|
2015-09-18 20:11:58 +02:00
|
|
|
|
|
|
|
|
2016-08-22 07:40:24 +02:00
|
|
|
@pytest.fixture(autouse=True)
|
2017-09-19 10:35:54 +02:00
|
|
|
def server_after_test(server, request):
|
|
|
|
"""Fixture to clean server request list after each test."""
|
|
|
|
request.node._server_log = server.captured_log
|
2015-09-18 20:11:58 +02:00
|
|
|
yield
|
2017-09-19 10:35:54 +02:00
|
|
|
server.after_test()
|
2016-01-12 22:48:38 +01:00
|
|
|
|
|
|
|
|
2016-08-22 07:40:24 +02:00
|
|
|
@pytest.fixture
|
2016-01-12 22:48:38 +01:00
|
|
|
def ssl_server(request, qapp):
|
|
|
|
"""Fixture for a webserver with a self-signed SSL certificate.
|
|
|
|
|
2017-09-19 10:35:54 +02:00
|
|
|
This needs to be explicitly used in a test, and overwrites the server log
|
2016-01-12 22:48:38 +01:00
|
|
|
used in that test.
|
|
|
|
"""
|
|
|
|
server = WebserverProcess('webserver_sub_ssl')
|
2017-09-19 10:35:54 +02:00
|
|
|
request.node._server_log = server.captured_log
|
2016-01-12 22:48:38 +01:00
|
|
|
server.start()
|
|
|
|
yield server
|
|
|
|
server.after_test()
|
|
|
|
server.cleanup()
|