Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

refactor(testing): deprecate testtools support in TestCase #2405

Draft
wants to merge 4 commits into
base: master
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions docs/_newsfragments/2156.breakingchange.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
- **Deprecation**: Support for `testtools` in `falcon.testing` is now deprecated and will be removed in Falcon 5.0.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Don't name the newsfragment .breakingchange., because we are not making any actual breaking changes until 5.0.

Since we are adding a new way to configure the base test case, I suggest marketing this as new feature.

If `testtools` is selected automatically (i.e., without explicit configuration via `FALCON_BASE_TEST_CASE`), a deprecation warning will be issued.
Explicit configuration using `FALCON_BASE_TEST_CASE=testtools.TestCase` will not emit a warning.
82 changes: 58 additions & 24 deletions falcon/testing/test_case.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,36 +18,79 @@
utilities for simulating and validating HTTP requests.
"""

try:
import testtools as unittest
except ImportError: # pragma: nocover
import unittest
import os
import unittest
import warnings

import falcon
import falcon.request

# TODO hoist for backwards compat. Remove in falcon 4.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please don't remove this note!
We missed it in Falcon 4.0 it seems, so it needs to be updated to say Falcon 5.0, if you want to touch it.

from falcon.testing.client import Result # NOQA
from falcon.testing.client import TestClient

base_case = os.environ.get('FALCON_BASE_TEST_CASE')

if base_case and 'testtools.TestCase' in base_case:
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We need to actually import the specified module, not just support testtools.TestCase.
It can be something.else.TestCase too.

try:
import testtools

Check warning on line 34 in falcon/testing/test_case.py

View check run for this annotation

Codecov / codecov/patch

falcon/testing/test_case.py#L33-L34

Added lines #L33 - L34 were not covered by tests

BaseTestCase = testtools.TestCase
except ImportError:
BaseTestCase = unittest.TestCase

Check warning on line 38 in falcon/testing/test_case.py

View check run for this annotation

Codecov / codecov/patch

falcon/testing/test_case.py#L36-L38

Added lines #L36 - L38 were not covered by tests
elif base_case is None:
try:
import testtools

warnings.warn(

Check warning on line 43 in falcon/testing/test_case.py

View check run for this annotation

Codecov / codecov/patch

falcon/testing/test_case.py#L43

Added line #L43 was not covered by tests
'Support for testtools is deprecated and will be removed in Falcon 5.0.',
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As requested before, please do not warn just because the package is importable!
Only warn when a TestCase is actually initialized.

DeprecationWarning,
)
BaseTestCase = testtools.TestCase

Check warning on line 47 in falcon/testing/test_case.py

View check run for this annotation

Codecov / codecov/patch

falcon/testing/test_case.py#L47

Added line #L47 was not covered by tests
except ImportError:
BaseTestCase = unittest.TestCase
else:
BaseTestCase = unittest.TestCase

Check warning on line 51 in falcon/testing/test_case.py

View check run for this annotation

Codecov / codecov/patch

falcon/testing/test_case.py#L51

Added line #L51 was not covered by tests


class TestCase(unittest.TestCase, TestClient):
"""Extends :mod:`unittest` to support WSGI/ASGI functional testing.
"""Extends ``unittest`` to support WSGI/ASGI functional testing.

Note:
If available, uses :mod:`testtools` in lieu of
:mod:`unittest`.
This class uses ``unittest`` by default, but you may use ``pytest``
to run ``unittest.TestCase`` instances, allowing a hybrid approach.

This base class provides some extra plumbing for unittest-style
test cases, to help simulate WSGI or ASGI requests without having
to spin up an actual web server. Various simulation methods are
derived from :class:`falcon.testing.TestClient`.
Recommended:
We recommend using **pytest** alongside **unittest** for testing.
See our tutorial on using pytest.

This base class provides extra functionality for unittest-style test cases,
helping simulate WSGI or ASGI requests without spinning up a web server. Various
simulation methods are derived from :class:`falcon.testing.TestClient`.

Simply inherit from this class in your test case classes instead of
:class:`unittest.TestCase` or :class:`testtools.TestCase`.
``unittest.TestCase``.

For example::

from falcon import testing
import myapp

class MyTestCase(testing.TestCase):
def setUp(self):
super(MyTestCase, self).setUp()

# Assume the hypothetical `myapp` package has a
# function called `create()` to initialize and
# return a `falcon.App` instance.
self.app = myapp.create()

class TestMyApp(MyTestCase):
def test_get_message(self):
doc = {'message': 'Hello world!'}

result = self.simulate_get('/messages/42')
self.assertEqual(result.json, doc)
"""

# NOTE(vytas): Here we have to restore __test__ to allow collecting tests!
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why have you removed my note?

__test__ = True

app: falcon.App
Expand All @@ -60,7 +103,6 @@
from falcon import testing
import myapp


class MyTestCase(testing.TestCase):
def setUp(self):
super(MyTestCase, self).setUp()
Expand All @@ -69,14 +111,6 @@
# function called `create()` to initialize and
# return a `falcon.App` instance.
self.app = myapp.create()


class TestMyApp(MyTestCase):
def test_get_message(self):
doc = {'message': 'Hello world!'}

result = self.simulate_get('/messages/42')
self.assertEqual(result.json, doc)
"""

def setUp(self) -> None:
Expand Down