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 all 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.newandimproved.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.
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.
84 changes: 65 additions & 19 deletions falcon/testing/test_case.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,33 +18,77 @@
utilities for simulating and validating HTTP requests.
"""

from importlib import import_module
import os
import warnings

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

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

# TODO hoist for backwards compat. Remove in falcon 5.

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

if base_case_path:
try:
module_path, class_name = base_case_path.rsplit('.', 1)
module = import_module(module_path)
BaseTestCase = getattr(module, class_name)
except (ImportError, AttributeError):
warnings.warn(
f"Could not import '{base_case_path}', defaulting to testtools.TestCase.",
ImportWarning,
)
BaseTestCase = unittest.TestCase
else:
BaseTestCase = unittest.TestCase


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?

Expand All @@ -60,7 +104,6 @@ class TestCase(unittest.TestCase, TestClient):
from falcon import testing
import myapp


class MyTestCase(testing.TestCase):
def setUp(self):
super(MyTestCase, self).setUp()
Expand All @@ -69,19 +112,22 @@ def setUp(self):
# 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:
super(TestCase, self).setUp()

if (
BaseTestCase == unittest.TestCase
and os.environ.get('FALCON_BASE_TEST_CASE') is None
and 'testtools' in str(unittest)
):
warnings.warn(
'Support for testtools is deprecated'
'and will be removed in Falcon 5.0.',
DeprecationWarning,
)

app = falcon.App()

# NOTE(kgriffs): Don't use super() to avoid triggering
Expand Down