Skip to content

Commit e9b8321

Browse files
northernSagejeff1evesqueGabriel Fioravante
authored
Extend test suite and remove deprecated code (#133)
* #45: features.rst, add 'HTTP Request' section * #45: features.rst, provide alternative, self contained example * #45: features.rst, remove dummy data * #45: features.rst, add newline at start of code example * #45: features.rst, reduce example syntax * update coverage run instructions to use pytest-cov * add changelog entry * use pytest command instead of py.test * move deprecation decorator to new _internal.py module * update pre-commit hooks * move LiveServer class from fixtures.py to a separate module * move helper functions to _internal.py * update tox.ini and coverage instructions to include imports * ommit unused port variable * move json response tests to separate module * remove philosophical contemplation of existence * remove deprecation test * remove test for deprecated method * remove deprecated LiveServer.url * update coverage instructions to account for multiprocessing * add .swp files to gitignore * add new tests for fixtures and deprecation decorator * add test for possible exception in liveserver._stop_cleanly() * remove pytest-cov * update for werkzeug 2.0.0 * fix deprecation warning * remove old py2 code * remove JSONResponse.json * adjust test for response overwriting * skip coverage on old compat function * update docs * update changelog * fix doc typos Co-authored-by: jeff levesque <[email protected]> Co-authored-by: Gabriel Fioravante <[email protected]>
1 parent 9f81e81 commit e9b8321

20 files changed

+290
-230
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
build
55
sdist
66
.vscode
7+
*.swp
78
__pycache__/
89

910
# virtualenv

.pre-commit-config.yaml

+7-3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
repos:
22
- repo: https://github.com/pre-commit/pre-commit-hooks
3-
rev: v3.2.0
3+
rev: v3.4.0
44
hooks:
55
- id: check-byte-order-marker
66
- id: trailing-whitespace
@@ -9,7 +9,7 @@ repos:
99
args: [--remove]
1010
- id: check-yaml
1111
- repo: https://github.com/asottile/reorder_python_imports
12-
rev: v2.3.5
12+
rev: v2.4.0
1313
hooks:
1414
- id: reorder-python-imports
1515
args: ['--application-directories=.:src', --py3-plus]
@@ -18,7 +18,7 @@ repos:
1818
hooks:
1919
- id: black
2020
- repo: https://gitlab.com/pycqa/flake8
21-
rev: 3.8.3
21+
rev: 3.8.4
2222
hooks:
2323
- id: flake8
2424
additional_dependencies: [flake8-bugbear]
@@ -30,3 +30,7 @@ repos:
3030
files: ^(HOWTORELEASE.rst|README.rst)$
3131
language: python
3232
additional_dependencies: [pygments, restructuredtext_lint]
33+
- repo: https://github.com/myint/autoflake.git
34+
rev: v1.4
35+
hooks:
36+
- id: autoflake

CONTRIBUTING.rst

+2-2
Original file line numberDiff line numberDiff line change
@@ -102,8 +102,8 @@ To get a complete report of code sections not being touched by the
102102
test suite run ``pytest`` using ``coverage``.
103103

104104
.. code-block:: text
105-
106-
$ coverage run -m pytest
105+
$ coverage run --concurrency=multiprocessing -m pytest
106+
$ coverage combine
107107
$ coverage html
108108
109109
Open ``htmlcov/index.html`` in your browser.

docs/changelog.rst

+7
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,13 @@
33
Changelog
44
=========
55

6+
1.2.0 (UNRELEASED)
7+
------------------
8+
9+
- Remove deprecated ``:meth:live_server.url``
10+
- fixture ``request_ctx is now deprecated``
11+
and will be removed in the future
12+
613
1.1.0 (2020-11-08)
714
------------------
815

docs/features.rst

+10-10
Original file line numberDiff line numberDiff line change
@@ -27,8 +27,8 @@ Extension provides some sugar for your tests, such as:
2727
2828
.. note::
2929

30-
User-defined ``json`` attribute/method in application response class does
31-
not overrides. So you can define your own response deserialization method:
30+
User-defined ``json`` attribute/method in application response class will
31+
not be overwritten. So you can define your own response deserialization method:
3232

3333
.. code:: python
3434
@@ -83,9 +83,9 @@ An instance of ``app.test_client``. Typically refers to
8383

8484
.. hint::
8585

86-
During tests execution the request context has been pushed, e.g.
87-
``url_for``, ``session`` and other context bound objects are available
88-
without context managers.
86+
During test execution a request context will be automatically pushed
87+
for you, so context-bound methods can be conveniently called (e.g.
88+
``url_for``, ``session``.
8989

9090
Example:
9191

@@ -145,7 +145,7 @@ other headless browsers).
145145
``--no-start-live-server`` - don’t start live server automatically
146146
``````````````````````````````````````````````````````````````````
147147

148-
By default the server is starting automatically whenever you reference
148+
By default the server will start automatically whenever you reference
149149
``live_server`` fixture in your tests. But starting live server imposes some
150150
high costs on tests that need it when they may not be ready yet. To prevent
151151
that behaviour pass ``--no-start-live-server`` into your default options (for
@@ -188,9 +188,11 @@ in your project's ``pytest.ini`` file)::
188188
addopts = --live-server-port=5000
189189

190190

191-
``request_ctx`` - request context
191+
``request_ctx`` - request context (Deprecated)
192192
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
193193

194+
**This fixture is deprecated and will be removed in the future.**
195+
194196
The request context which contains all request relevant information.
195197

196198
.. hint::
@@ -228,7 +230,7 @@ Common request methods are available through the internals of the `Flask API`_.
228230
Specifically, the API creates the default `flask.Flask.test_client`_ instance,
229231
which works like a regular `Werkzeug test client`_.
230232

231-
Example:
233+
Examples:
232234

233235
.. code:: python
234236
@@ -247,8 +249,6 @@ Example:
247249
248250
assert res.status_code == 200
249251
250-
Example:
251-
252252
.. code:: python
253253
254254
def test_get_request(client, live_server):

docs/tutorial.rst

+2-2
Original file line numberDiff line numberDiff line change
@@ -31,9 +31,9 @@ Define your application fixture in ``conftest.py``:
3131
Step 3. Run your test suite
3232
---------------------------
3333

34-
Use the ``py.test`` command to run your test suite::
34+
Use the ``pytest`` command to run your test suite::
3535

36-
py.test
36+
pytest
3737

3838
.. note:: Test discovery.
3939

pytest_flask/_internal.py

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
import functools
2+
import warnings
3+
4+
5+
def deprecated(reason):
6+
"""Decorator which can be used to mark function or method as deprecated.
7+
It will result a warning being emmitted when the function is called."""
8+
9+
def decorator(func):
10+
@functools.wraps(func)
11+
def deprecated_call(*args, **kwargs):
12+
warnings.simplefilter("always", DeprecationWarning)
13+
warnings.warn(reason, DeprecationWarning, stacklevel=2)
14+
warnings.simplefilter("default", DeprecationWarning)
15+
return func(*args, **kwargs)
16+
17+
return deprecated_call
18+
19+
return decorator
20+
21+
22+
def _rewrite_server_name(server_name, new_port):
23+
"""Rewrite server port in ``server_name`` with ``new_port`` value."""
24+
sep = ":"
25+
if sep in server_name:
26+
server_name, _ = server_name.split(sep, 1)
27+
return sep.join((server_name, new_port))
28+
29+
30+
def _determine_scope(*, fixture_name, config):
31+
return config.getini("live_server_scope")
32+
33+
34+
def _make_accept_header(mimetype):
35+
return [("Accept", mimetype)]

pytest_flask/fixtures.py

+15-135
Original file line numberDiff line numberDiff line change
@@ -1,33 +1,15 @@
11
#!/usr/bin/env python
2-
import functools
3-
import logging
4-
import multiprocessing
5-
import os
6-
import signal
72
import socket
8-
import time
93
import warnings
104

115
import pytest
126
from flask import _request_ctx_stack
137

14-
15-
def deprecated(reason):
16-
"""Decorator which can be used to mark function or method as deprecated.
17-
It will result a warning being emmitted when the function is called.
18-
"""
19-
20-
def decorator(func):
21-
@functools.wraps(func)
22-
def deprecated_call(*args, **kwargs):
23-
warnings.simplefilter("always", DeprecationWarning)
24-
warnings.warn(reason, DeprecationWarning, stacklevel=2)
25-
warnings.simplefilter("default", DeprecationWarning)
26-
return func(*args, **kwargs)
27-
28-
return deprecated_call
29-
30-
return decorator
8+
from ._internal import _determine_scope
9+
from ._internal import _make_accept_header
10+
from ._internal import _rewrite_server_name
11+
from ._internal import deprecated
12+
from .live_server import LiveServer
3113

3214

3315
@pytest.fixture
@@ -58,114 +40,7 @@ def test_login(self):
5840
request.cls.client = client
5941

6042

61-
class LiveServer:
62-
"""The helper class used to manage a live server. Handles creation and
63-
stopping application in a separate process.
64-
65-
:param app: The application to run.
66-
:param host: The host where to listen (default localhost).
67-
:param port: The port to run application.
68-
:param wait: The timeout after which test case is aborted if
69-
application is not started.
70-
"""
71-
72-
def __init__(self, app, host, port, wait, clean_stop=False):
73-
self.app = app
74-
self.port = port
75-
self.host = host
76-
self.wait = wait
77-
self.clean_stop = clean_stop
78-
self._process = None
79-
80-
def start(self):
81-
"""Start application in a separate process."""
82-
83-
def worker(app, host, port):
84-
app.run(host=host, port=port, use_reloader=False, threaded=True)
85-
86-
self._process = multiprocessing.Process(
87-
target=worker, args=(self.app, self.host, self.port)
88-
)
89-
self._process.daemon = True
90-
self._process.start()
91-
92-
keep_trying = True
93-
start_time = time.time()
94-
while keep_trying:
95-
elapsed_time = time.time() - start_time
96-
if elapsed_time > self.wait:
97-
pytest.fail(
98-
"Failed to start the server after {!s} "
99-
"seconds.".format(self.wait)
100-
)
101-
if self._is_ready():
102-
keep_trying = False
103-
104-
def _is_ready(self):
105-
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
106-
try:
107-
sock.connect((self.host, self.port))
108-
except socket.error:
109-
ret = False
110-
else:
111-
ret = True
112-
finally:
113-
sock.close()
114-
return ret
115-
116-
@deprecated(
117-
reason=(
118-
'The "live_server.url" method is deprecated and will '
119-
"be removed in the future. Please use "
120-
'the "flask.url_for" function instead.',
121-
)
122-
)
123-
def url(self, url=""):
124-
"""Returns the complete url based on server options."""
125-
return "http://{host!s}:{port!s}{url!s}".format(
126-
host=self.host, port=self.port, url=url
127-
)
128-
129-
def stop(self):
130-
"""Stop application process."""
131-
if self._process:
132-
if self.clean_stop and self._stop_cleanly():
133-
return
134-
if self._process.is_alive():
135-
# If it's still alive, kill it
136-
self._process.terminate()
137-
138-
def _stop_cleanly(self, timeout=5):
139-
"""Attempts to stop the server cleanly by sending a SIGINT signal and waiting for
140-
``timeout`` seconds.
141-
142-
:return: True if the server was cleanly stopped, False otherwise.
143-
"""
144-
try:
145-
os.kill(self._process.pid, signal.SIGINT)
146-
self._process.join(timeout)
147-
return True
148-
except Exception as ex:
149-
logging.error("Failed to join the live server process: %r", ex)
150-
return False
151-
152-
def __repr__(self):
153-
return "<LiveServer listening at %s>" % self.url()
154-
155-
156-
def _rewrite_server_name(server_name, new_port):
157-
"""Rewrite server port in ``server_name`` with ``new_port`` value."""
158-
sep = ":"
159-
if sep in server_name:
160-
server_name, port = server_name.split(sep, 1)
161-
return sep.join((server_name, new_port))
162-
163-
164-
def determine_scope(*, fixture_name, config):
165-
return config.getini("live_server_scope")
166-
167-
168-
@pytest.fixture(scope=determine_scope)
43+
@pytest.fixture(scope=_determine_scope)
16944
def live_server(request, app, pytestconfig):
17045
"""Run application in a separate process.
17146
@@ -223,6 +98,15 @@ def request_ctx(app):
22398
"""The request context which contains all request relevant information,
22499
e.g. `session`, `g`, `flashes`, etc.
225100
"""
101+
warnings.warn(
102+
"In Werzeug 2.0.0, the Client request methods "
103+
"(client.get, client.post) always return an instance of TestResponse. This "
104+
"class provides a reference to the request object through 'response.request' "
105+
"The fixture 'request_ctx' is deprecated and will be removed in the future, using TestResponse.request "
106+
"is the prefered way.",
107+
DeprecationWarning,
108+
stacklevel=2,
109+
)
226110
return _request_ctx_stack.top
227111

228112

@@ -231,10 +115,6 @@ def mimetype(request):
231115
return request.param
232116

233117

234-
def _make_accept_header(mimetype):
235-
return [("Accept", mimetype)]
236-
237-
238118
@pytest.fixture
239119
def accept_mimetype(mimetype):
240120
return _make_accept_header(mimetype)

0 commit comments

Comments
 (0)