diff --git a/.github/workflows/test-other.yaml b/.github/workflows/test-other.yaml new file mode 100644 index 000000000..4e8289263 --- /dev/null +++ b/.github/workflows/test-other.yaml @@ -0,0 +1,48 @@ +# Other tests, primarily those requiring obsolete dependencies. +name: test-other + +on: + push: + branches: + - "*" + +jobs: + run-tox: + name: tox -e ${{ matrix.toxenv }} + runs-on: ubuntu-20.04 + strategy: + fail-fast: false + matrix: + python-version: + - "3.8" + toxenv: + - "py38_smoke" + - "py38_smoke_cython" + - "wsgi_meinheld" + + steps: + - name: Checkout repo + uses: actions/checkout@v4 + with: + fetch-depth: 2 + + - name: Set up Python + uses: actions/setup-python@v5 + with: + python-version: ${{ matrix.python-version }} + + - name: Install smoke test dependencies + run: | + sudo apt-get update + sudo apt-get install -y libunwind-dev + + - name: Install Python dependencies + run: | + python -m pip install --upgrade pip + pip install --upgrade setuptools tox wheel + python --version + pip --version + tox --version + + - name: Run tox + run: tox -e ${{ matrix.toxenv }} diff --git a/.github/workflows/test-wheels.yaml b/.github/workflows/test-wheels.yaml index 3ccf2e360..43c7644a7 100644 --- a/.github/workflows/test-wheels.yaml +++ b/.github/workflows/test-wheels.yaml @@ -22,6 +22,8 @@ jobs: - build: "cp313-manylinux_s390x" os: ubuntu-latest emulation: true + - build: "cp38-manylinux_x86_64" + os: ubuntu-latest - build: "cp313-musllinux_x86_64" os: ubuntu-latest - build: "cp313-macosx_arm64" diff --git a/.github/workflows/tests.yaml b/.github/workflows/tests.yaml index 078e5f59a..47ece6bef 100644 --- a/.github/workflows/tests.yaml +++ b/.github/workflows/tests.yaml @@ -1,4 +1,5 @@ -name: Run tests +# Run tests in various tox environments. +name: tests on: # NOTE(vytas): Trigger the tests workflow on push or pull request @@ -11,104 +12,73 @@ on: - master jobs: - run_tox: - name: tox -e ${{ matrix.toxenv }} (${{matrix.python-version}} on ${{ matrix.os }}) - runs-on: ${{ matrix.os }} + run-tox: + name: tox -e ${{ matrix.tox.env }}${{ matrix.tox.platform-label || '' }} + runs-on: ${{ matrix.tox.os || matrix.default-os }} strategy: fail-fast: false matrix: + default-os: + - "ubuntu-latest" python-version: - "3.12" - python-dev-version: - - "" - os: - - "ubuntu-latest" - toxenv: - - "pep8" - - "pep8-docstrings" - - "pep8-examples" - - "ruff" - - "mypy" - - "mypy_tests" - - "py312" - - "py312_sans_msgpack" - - "py312_cython" - - "docs" - - "towncrier" - - "look" - - "asgilook" - - "ws_tutorial" - # NOTE(vytas): Temporarily disabled while we are releasing a - # new version of python-mimeparse, and haven't decided yet whether - # to keep vendoring or reimplement. - # - "check_vendored" - - "twine_check" - - "daphne" - - "hypercorn" - - "e2e_chrome" - - "e2e_firefox" - - "no_optional_packages" - # TODO(kgriffs): Re-enable once hug has a chance to address - # breaking changes in Falcon 3.0 - # - "hug" - include: - - python-version: pypy3.9 - os: ubuntu-latest - toxenv: pypy3 - - python-version: "3.8" - os: ubuntu-latest - toxenv: py38 - - python-version: "3.9" - os: ubuntu-latest - toxenv: py39 - - python-version: "3.9" - os: ubuntu-latest - toxenv: py39_cython - - python-version: "3.10" - os: ubuntu-latest - toxenv: py310 - - python-version: "3.10" - os: ubuntu-latest - toxenv: py310_cython - - python-version: "3.11" - os: ubuntu-latest - toxenv: py311 - - python-version: "3.11" - os: ubuntu-latest - toxenv: py311_cython - - python-version: "3.12" - os: ubuntu-latest - toxenv: py312 - - python-version: "3.12" - os: ubuntu-latest - toxenv: py312_cython - - python-version: "3.12" + tox: + # Lint + - env: pep8 + - env: pep8-docstrings + - env: ruff + # Documentation + - env: docs + - env: towncrier + # Typing + - env: mypy + - env: mypy_tests + # Python tests + - env: mintest + coverage: true + - env: pypy3 + python-version: "pypy3.10" + - env: py38 + python-version: "3.8" + - env: py39 + python-version: "3.9" + - env: py310 + python-version: "3.10" + - env: py310_cython + python-version: "3.10" + - env: py311 + python-version: "3.11" + - env: py311_cython + python-version: "3.11" + - env: py312 + coverage: true + python-version: "3.12" + - env: py312_cython + python-version: "3.12" + - env: py313 + python-version: "3.13.0-rc.3 - 3.13" + - env: py313_cython + python-version: "3.13.0-rc.3 - 3.13" + - env: py312_nocover os: macos-latest - toxenv: py312_nocover - - python-version: "3.12" + platform-label: ' (macos)' + - env: py312_nocover os: windows-latest - toxenv: py312_nocover - - python-version: "3.13" - python-dev-version: "3.13.0-rc.1 - 3.13" - os: ubuntu-latest - toxenv: py313 - - python-version: "3.13" - python-dev-version: "3.13.0-rc.1 - 3.13" - os: ubuntu-latest - toxenv: py313_cython - # These env require 3.8 and 20.04, see tox.ini - - python-version: "3.8" - os: ubuntu-20.04 - toxenv: py38_smoke - - python-version: "3.8" - os: ubuntu-20.04 - toxenv: py38_smoke_cython - - python-version: "3.8" - os: ubuntu-latest - toxenv: "wsgi_servers" + platform-label: ' (windows)' + # Tutorials + - env: "look" + - env: "asgilook" + - env: "ws_tutorial" + # Tooling + - env: "twine_check" + # ASGI & WSGI servers + - env: "daphne" + - env: "hypercorn" + - env: "wsgi_servers" + # E2E tests + - env: "e2e_chrome" + - env: "e2e_firefox" - # Steps to run in each job. - # Some are GitHub actions, others run shell commands. steps: - name: Checkout repo uses: actions/checkout@v4 @@ -119,43 +89,30 @@ jobs: - name: Set up Python uses: actions/setup-python@v5 - if: ${{ !matrix.python-dev-version }} with: - python-version: ${{ matrix.python-version }} - - - name: Set up Python (pre-release) - uses: actions/setup-python@v5 - if: ${{ matrix.python-dev-version }} - with: - python-version: ${{ matrix.python-dev-version }} - - - name: Install smoke test dependencies - if: ${{ matrix.toxenv == 'py38_smoke' || matrix.toxenv == 'py38_smoke_cython' }} - run: | - sudo apt-get update - sudo apt-get install -y libunwind-dev + python-version: ${{ matrix.tox.python-version || matrix.python-version }} - - name: Install dependencies + - name: Install Python dependencies run: | python -m pip install --upgrade pip - pip install -U coverage fixtures setuptools tox wheel + pip install --upgrade coverage setuptools tox wheel python --version pip --version tox --version coverage --version - - name: Run tests - run: tox -e ${{ matrix.toxenv }} + - name: Run tox + run: tox -e ${{ matrix.tox.env }} - name: Combine coverage - if: ${{ matrix.toxenv == 'py312' || matrix.toxenv == 'py312_sans_msgpack' }} + if: ${{ matrix.tox.coverage }} run: | coverage --version coverage combine - name: Upload coverage to Codecov uses: codecov/codecov-action@v4 - if: ${{ matrix.toxenv == 'py312' || matrix.toxenv == 'py312_sans_msgpack' }} + if: ${{ matrix.tox.coverage }} with: env_vars: PYTHON fail_ci_if_error: true diff --git a/.github/workflows/mintest.yaml b/.github/workflows/tox-sdist.yaml similarity index 96% rename from .github/workflows/mintest.yaml rename to .github/workflows/tox-sdist.yaml index c3a6b5f78..0986fc841 100644 --- a/.github/workflows/mintest.yaml +++ b/.github/workflows/tox-sdist.yaml @@ -1,4 +1,4 @@ -name: Run tests (contributor's checklist) +name: test-tox-sdist on: # Trigger the workflow on master but also allow it to run manually. diff --git a/AUTHORS b/AUTHORS index 3ec67da81..95b44b1fd 100644 --- a/AUTHORS +++ b/AUTHORS @@ -23,8 +23,10 @@ listed below by date of first contribution: * Derrick Wippler (thrawn01) * Sebastián Magrí (sebasmagri) * Chris Petersen (ex-nerd) +* Cindy Pallares (cindy) * Andy Freeland (rouge8) * Michał Jaworski (swistakm) +* Michele Lacchia (rubik) * Jason Campbell (xiaclo) * Barry Warsaw (warsaw) * Jeffrey Zhang (jeffrey4l) @@ -34,6 +36,7 @@ listed below by date of first contribution: * David Pineau (Joacchim) * Benjamen Meyer (BenjamenMeyer) * Sriram Madapusi Vasudevan (TheSriram) +* Omer Katz (thedrow) * Erik Erwitt (eerwitt) * Bernhard Weitzhofer (b6d) * Rahman Syed (rsyed83) @@ -49,6 +52,8 @@ listed below by date of first contribution: * Maxim Tsoy (muodov) * Christian Pedersen (chripede) * Harrison Pincket (hpincket) +* Richard Olsson (richardolsson) +* Zack Gold (zg) * Usman Ehtesham Gul (ueg1990) * Adam Yala (adamyala) * Grigory Bakunov (bobuk) @@ -70,7 +75,58 @@ listed below by date of first contribution: * Yash Mehrotra (yashmehrotra) * Stephen Milner (ashcrow) * ksonj +* Christian Jensen (jensenbox) +* Sam Raker (swizzard) +* Ali-Akber Saifee (alisaifee) +* stiefel40k +* Zhigang Wang (w1z2g3) +* Rafał Kasa (rafalkasa) +* Qingping Hou (houqp) +* wssbck +* Martin Asell Backlund (masellfoodpanda) +* Matt Giles (mattgiles) +* Naveen Yadav (navyad) +* roeih-stratoscale +* Jahongir (rahmonov) +* Hans Lellelid (hozn) +* Steven Seguin (sseg) +* Michal Bultrowicz (butla) +* Joshua Adelman (synapticarbors) +* Ron Rothman (RonRothman) +* mayurmahajan +* Serge (yasek) +* Munir Contractor (munircontractor) +* Max (DSIW) +* Caleb Hayashida (santeyio) +* Tim Chen (timc13) +* Griff George (GriffGeorge) +* Russ (rhemz) +* Hynek Schlawack (hynek) +* Tyrone Groves (tyronegroves) +* Justin Siebert (Just-Sieb) * Vytautas Liuolia (vytas7) +* Hugo van Kemenade (hugovk) +* Ryan Turner (ZDBioHazard) +* Federico Caselli (CaselIT) +* Steven Jackson (steven-upside) +* ozzzik +* elf69 (selfvin) +* Josh Klar (klardotsh) +* Carlo (carlodri) +* Akeem King (KingAkeem) +* Dmitrii Trofimov (DmitriiTrofimov) +* Anthony Bush (awbush) +* bkcsfi +* Aston Motes (astonm) +* TheMushr00m (TheMushrr00m) +* Aly Sivji (alysivji) +* Anthony (tribals) +* Alan Parker Lue (aparkerlue) +* Nate D. (Nateyo) +* Rajan Paneru (paneru-rajan) +* Dmitri Vasilishin (dmvass) +* Andrii Marynets (amarynets) +* Zubair Akram (zubairakram) * Nick Zaccardi (nZac) * Safa Ozturk (safaozturk93) * Brent Smith (edmondb) @@ -90,7 +146,6 @@ listed below by date of first contribution: * Mostafa Khaki (mosi-kha) * nagaabhinaya * Matt Donders (mattdonders) -* Federico Caselli (CaselIT) * Yang Keming (kemingy) * Tim Gates (timgates42) * Paul (hackedd) @@ -143,6 +198,7 @@ listed below by date of first contribution: * Dave Tapley (davetapley) * Agustin Arce (aarcex3) * Christian Grossmüller (chgad) +* Sai Prathik R (prathik2401) (et al.) diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 5422355cc..0307fb191 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -37,6 +37,14 @@ $ pip install -U ruff $ ruff format ``` +You can also reformat your code, and apply safe ``ruff`` fixes, via the +``reformat`` ``tox`` environment: + +```bash +$ pip install -U tox +$ tox -e reformat +``` + You can check all this by running ``tox`` from within the Falcon project directory. Your environment must be based on CPython 3.10, 3.11, 3.12 or 3.13: diff --git a/README.rst b/README.rst index 996555128..2e5d1b3fc 100644 --- a/README.rst +++ b/README.rst @@ -4,7 +4,7 @@ :target: https://falconframework.org/ :width: 100 % -|Build Status| |Docs| |codecov.io| +|Build status| |Docs| |codecov.io| |PyPI package| |Python versions| The Falcon Web Framework ======================== @@ -278,7 +278,7 @@ available to your app without having to reinstall the package: .. code:: bash $ cd falcon - $ pip install -e . + $ FALCON_DISABLE_CYTHON=Y pip install -e . You can manually test changes to the Falcon framework by switching to the directory of the cloned repo and then running pytest: @@ -978,13 +978,17 @@ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. +.. |Build status| image:: https://github.com/falconry/falcon/actions/workflows/tests.yaml/badge.svg + :target: https://github.com/falconry/falcon/actions/workflows/tests.yaml .. |Docs| image:: https://readthedocs.org/projects/falcon/badge/?version=stable :alt: Falcon web framework docs :target: https://falcon.readthedocs.io/en/stable/?badge=stable -.. |Build Status| image:: https://github.com/falconry/falcon/workflows/Run%20tests/badge.svg - :target: https://github.com/falconry/falcon/actions?query=workflow%3A%22Run+tests%22 .. |codecov.io| image:: https://codecov.io/gh/falconry/falcon/branch/master/graphs/badge.svg - :target: http://codecov.io/gh/falconry/falcon + :target: https://codecov.io/gh/falconry/falcon +.. |PyPI package| image:: https://badge.fury.io/py/falcon.svg + :target: https://pypi.org/project/falcon/ +.. |Python versions| image:: https://img.shields.io/pypi/pyversions/falcon.svg + :target: https://pypi.org/project/falcon/ .. |Backer:GovCert| image:: https://falconframework.org/assets/govcert.png :alt: CERT Gouvernemental Luxembourg :height: 60px diff --git a/docs/_newsfragments/1367.newandimproved.rst b/docs/_newsfragments/1367.newandimproved.rst new file mode 100644 index 000000000..e4b4e5c09 --- /dev/null +++ b/docs/_newsfragments/1367.newandimproved.rst @@ -0,0 +1,3 @@ +The new implementation of :ref:`media type utilities ` +(Falcon was using the ``python-mimeparse`` library before) now always favors +the exact media type match, if one is available. diff --git a/docs/_newsfragments/864.breakingchange.rst b/docs/_newsfragments/864.breakingchange.rst new file mode 100644 index 000000000..0c518bc1c --- /dev/null +++ b/docs/_newsfragments/864.breakingchange.rst @@ -0,0 +1,36 @@ +Falcon is no longer vendoring the +`python-mimeparse `__ library; +the relevant functionality has instead been reimplemented in the framework +itself, fixing a handful of long-standing bugs in the new implementation. + +If you use standalone +`python-mimeparse `__ in your +project, do not worry! We will continue to maintain it as a separate package +under the Falconry umbrella (we took over about 3 years ago). + +The following new behaviors are considered breaking changes: + +* Previously, the iterable passed to + :meth:`req.client_prefers ` had to be sorted in + the order of increasing desirability. + :func:`~falcon.mediatypes.best_match`, and by proxy + :meth:`~falcon.Request.client_prefers`, now consider the provided media types + to be sorted in the (more intuitive, we hope) order of decreasing + desirability. + +* Unlike ``python-mimeparse``, the new + :ref:`media type utilities ` consider media types with + different values for the same parameters as non-matching. + + One theoretically possible scenario where this change can affect you is only + installing a :ref:`media ` handler for a content type with parameters; + it then may not match media types with conflicting values (that used to match + before Falcon 4.0). + If this turns out to be the case, also + :ref:`install the same handler ` for the generic + ``type/subtype`` without parameters. + +The new functions, +:func:`falcon.mediatypes.quality` and :func:`falcon.mediatypes.best_match`, +otherwise have the same signature as the corresponding methods from +``python-mimeparse``. diff --git a/docs/api/util.rst b/docs/api/util.rst index f15a3b444..254b4cb60 100644 --- a/docs/api/util.rst +++ b/docs/api/util.rst @@ -33,10 +33,14 @@ HTTP Status .. autofunction:: falcon.http_status_to_code .. autofunction:: falcon.code_to_http_status +.. _mediatype_util: + Media types ----------- .. autofunction:: falcon.parse_header +.. autofunction:: falcon.mediatypes.quality +.. autofunction:: falcon.mediatypes.best_match Async ----- diff --git a/docs/changes/4.0.0.rst b/docs/changes/4.0.0.rst index f04f2c063..dea98a584 100644 --- a/docs/changes/4.0.0.rst +++ b/docs/changes/4.0.0.rst @@ -101,7 +101,9 @@ Many thanks to all of our talented and stylish contributors for this release! - `mgorny `__ - `mihaitodor `__ - `MRLab12 `__ +- `myusko `__ - `nfsec `__ +- `prathik2401 `__ - `RioAtHome `__ - `TigreModerata `__ - `vgerak `__ diff --git a/docs/ext/rfc.py b/docs/ext/rfc.py index abe13499b..3633d6ce0 100644 --- a/docs/ext/rfc.py +++ b/docs/ext/rfc.py @@ -22,6 +22,7 @@ import re +IETF_DOCS = 'https://datatracker.ietf.org/doc/html' RFC_PATTERN = re.compile(r'RFC (\d{4}), Section ([\d\.]+)') @@ -39,11 +40,10 @@ def _process_line(line): section = m.group(2) template = ( - '`RFC {rfc}, Section {section} ' - '`_' + '`RFC {rfc}, Section {section} <{ietf_docs}/rfc{rfc}#section-{section}>`__' ) - rendered_text = template.format(rfc=rfc, section=section) + rendered_text = template.format(rfc=rfc, section=section, ietf_docs=IETF_DOCS) return line[: m.start()] + rendered_text + line[m.end() :] diff --git a/docs/index.rst b/docs/index.rst index 359062b5f..329659791 100644 --- a/docs/index.rst +++ b/docs/index.rst @@ -16,26 +16,8 @@ We like to think of Falcon as the *Dieter Rams* of web frameworks. Falcon encourages the REST architectural style, and tries to do as little as possible while remaining highly effective. -.. code:: python - - import falcon - - class QuoteResource: - def on_get(self, req, resp): - """Handles GET requests""" - quote = { - 'quote': ( - "I've always been more interested in " - "the future than in the past." - ), - 'author': 'Grace Hopper' - } - - resp.media = quote - - - app = falcon.App() - app.add_route('/quote', QuoteResource()) +.. literalinclude:: ../examples/quote.py + :language: python For a fully working example, check out the :ref:`quickstart`. diff --git a/examples/quote.py b/examples/quote.py new file mode 100644 index 000000000..eb6a62469 --- /dev/null +++ b/examples/quote.py @@ -0,0 +1,14 @@ +import falcon + + +class QuoteResource: + def on_get(self, req: falcon.Request, resp: falcon.Response) -> None: + """Handle GET requests.""" + resp.media = { + 'quote': "I've always been more interested in the future than in the past.", + 'author': 'Grace Hopper', + } + + +app = falcon.App() +app.add_route('/quote', QuoteResource()) diff --git a/falcon/__init__.py b/falcon/__init__.py index 5c383b774..55ea9cde4 100644 --- a/falcon/__init__.py +++ b/falcon/__init__.py @@ -84,6 +84,7 @@ 'http_status_to_code', 'IS_64_BITS', 'is_python_func', + 'mediatypes', 'misc', 'parse_header', 'reader', @@ -138,6 +139,8 @@ 'HTTPUnsupportedMediaType', 'HTTPUriTooLong', 'HTTPVersionNotSupported', + 'InvalidMediaRange', + 'InvalidMediaType', 'MediaMalformedError', 'MediaNotFoundError', 'MediaValidationError', @@ -388,6 +391,8 @@ from falcon.errors import HTTPUnsupportedMediaType from falcon.errors import HTTPUriTooLong from falcon.errors import HTTPVersionNotSupported +from falcon.errors import InvalidMediaRange +from falcon.errors import InvalidMediaType from falcon.errors import MediaMalformedError from falcon.errors import MediaNotFoundError from falcon.errors import MediaValidationError @@ -617,6 +622,7 @@ from falcon.util import http_status_to_code from falcon.util import IS_64_BITS from falcon.util import is_python_func +from falcon.util import mediatypes from falcon.util import misc from falcon.util import parse_header from falcon.util import reader diff --git a/falcon/app_helpers.py b/falcon/app_helpers.py index 86654aa1a..d83731005 100644 --- a/falcon/app_helpers.py +++ b/falcon/app_helpers.py @@ -291,12 +291,14 @@ def default_serialize_error(req: Request, resp: Response, exception: HTTPError) resp: Instance of ``falcon.Response`` exception: Instance of ``falcon.HTTPError`` """ - predefined = [MEDIA_XML, 'text/xml', MEDIA_JSON] + + predefined = [MEDIA_JSON, 'text/xml', MEDIA_XML] media_handlers = [mt for mt in resp.options.media_handlers if mt not in predefined] - # NOTE(caselit) add all the registered before the predefined ones. This ensures that - # in case of equal match the last one (json) is selected and that the q= is taken - # into consideration when selecting the media - preferred = req.client_prefers(media_handlers + predefined) + # NOTE(caselit,vytas): Add the registered handlers after the predefined + # ones. This ensures that in the case of an equal match, the first one + # (JSON) is selected and that the q parameter is taken into consideration + # when selecting the media handler. + preferred = req.client_prefers(predefined + media_handlers) if preferred is None: # NOTE(kgriffs): See if the client expects a custom media diff --git a/falcon/asgi/app.py b/falcon/asgi/app.py index d28c8c49a..f3b637802 100644 --- a/falcon/asgi/app.py +++ b/falcon/asgi/app.py @@ -554,7 +554,7 @@ async def __call__( # type: ignore[override] # noqa: C901 data = resp._data if data is None and resp._media is not None: - # NOTE(kgriffs): We use a special MISSING singleton since + # NOTE(kgriffs): We use a special _UNSET singleton since # None is ambiguous (the media handler might return None). if resp._media_rendered is _UNSET: opt = resp.options diff --git a/falcon/asgi/request.py b/falcon/asgi/request.py index 8442ea8bc..ebd2e382d 100644 --- a/falcon/asgi/request.py +++ b/falcon/asgi/request.py @@ -169,7 +169,7 @@ def __init__( self.uri_template = None # PERF(vytas): Fall back to class variable(s) when unset. - # self._media = MISSING + # self._media = _UNSET # self._media_error = None # TODO(kgriffs): ASGI does not specify whether 'path' may be empty, diff --git a/falcon/asgi/response.py b/falcon/asgi/response.py index 42cc05830..1b5717444 100644 --- a/falcon/asgi/response.py +++ b/falcon/asgi/response.py @@ -207,7 +207,7 @@ async def render_body(self) -> Optional[bytes]: # type: ignore[override] data = self._data if data is None and self._media is not None: - # NOTE(kgriffs): We use a special MISSING singleton since + # NOTE(kgriffs): We use a special _UNSET singleton since # None is ambiguous (the media handler might return None). if self._media_rendered is _UNSET: if not self.content_type: diff --git a/falcon/errors.py b/falcon/errors.py index 4acc7247d..092fff2e5 100644 --- a/falcon/errors.py +++ b/falcon/errors.py @@ -88,6 +88,8 @@ def on_get(self, req, resp): 'HTTPUnsupportedMediaType', 'HTTPUriTooLong', 'HTTPVersionNotSupported', + 'InvalidMediaRange', + 'InvalidMediaType', 'MediaMalformedError', 'MediaNotFoundError', 'MediaValidationError', @@ -111,6 +113,14 @@ class CompatibilityError(ValueError): """The given method, value, or type is not compatible.""" +class InvalidMediaType(ValueError): + """The provided media type cannot be parsed into type/subtype.""" + + +class InvalidMediaRange(InvalidMediaType): + """The media range contains an invalid media type and/or the q value.""" + + class UnsupportedScopeError(RuntimeError): """The ASGI scope type is not supported by Falcon.""" diff --git a/falcon/http_error.py b/falcon/http_error.py index 508bd2524..b6c104aec 100644 --- a/falcon/http_error.py +++ b/falcon/http_error.py @@ -19,8 +19,7 @@ import xml.etree.ElementTree as et from falcon.constants import MEDIA_JSON -from falcon.util import code_to_http_status -from falcon.util import http_status_to_code +from falcon.util import misc from falcon.util import uri if TYPE_CHECKING: @@ -136,7 +135,7 @@ def __init__( # we'll probably switch over to making everything code-based to more # easily support HTTP/2. When that happens, should we continue to # include the reason phrase in the title? - self.title = title or code_to_http_status(status) + self.title = title or misc.code_to_http_status(status) self.description = description self.headers = headers @@ -161,7 +160,7 @@ def status_code(self) -> int: """HTTP status code normalized from the ``status`` argument passed to the initializer. """ # noqa: D205 - return http_status_to_code(self.status) + return misc.http_status_to_code(self.status) def to_dict( self, obj_type: Type[MutableMapping[str, Union[str, int, None, Link]]] = dict diff --git a/falcon/media/handlers.py b/falcon/media/handlers.py index 6b096ec91..b48a95d77 100644 --- a/falcon/media/handlers.py +++ b/falcon/media/handlers.py @@ -30,8 +30,8 @@ from falcon.media.multipart import MultipartFormHandler from falcon.media.multipart import MultipartParseOptions from falcon.media.urlencoded import URLEncodedFormHandler +from falcon.util import mediatypes from falcon.util import misc -from falcon.vendor import mimeparse class MissingDependencyHandler(BinaryBaseHandlerWS): @@ -186,7 +186,7 @@ def _best_match(media_type: str, all_media_types: Sequence[str]) -> Optional[str try: # NOTE(jmvrbanac): Mimeparse will return an empty string if it can # parse the media type, but cannot find a suitable type. - result = mimeparse.best_match(all_media_types, media_type) + result = mediatypes.best_match(all_media_types, media_type) except ValueError: pass diff --git a/falcon/request.py b/falcon/request.py index 9bc38a59b..64f9312e1 100644 --- a/falcon/request.py +++ b/falcon/request.py @@ -53,10 +53,10 @@ from falcon.typing import ReadableIO from falcon.util import deprecation from falcon.util import ETag +from falcon.util import mediatypes from falcon.util import structures from falcon.util.uri import parse_host from falcon.util.uri import parse_query_string -from falcon.vendor import mimeparse DEFAULT_ERROR_LOG_FORMAT = '{0:%Y-%m-%d %H:%M:%S} [FALCON] [ERROR] {1} {2}{3} => ' @@ -1167,7 +1167,7 @@ def client_accepts(self, media_type: str) -> bool: # Fall back to full-blown parsing try: - return mimeparse.quality(media_type, accept) != 0.0 + return mediatypes.quality(media_type, accept) != 0.0 except ValueError: return False @@ -1187,7 +1187,7 @@ def client_prefers(self, media_types: Iterable[str]) -> Optional[str]: try: # NOTE(kgriffs): best_match will return '' if no match is found - preferred_type = mimeparse.best_match(media_types, self.accept) + preferred_type = mediatypes.best_match(media_types, self.accept) except ValueError: # Value for the accept header was not formatted correctly preferred_type = '' diff --git a/falcon/response.py b/falcon/response.py index d468b0db3..e71a4b4c9 100644 --- a/falcon/response.py +++ b/falcon/response.py @@ -276,7 +276,7 @@ def render_body(self) -> Optional[bytes]: data = self._data if data is None and self._media is not None: - # NOTE(kgriffs): We use a special MISSING singleton since + # NOTE(kgriffs): We use a special _UNSET singleton since # None is ambiguous (the media handler might return None). if self._media_rendered is _UNSET: if not self.content_type: diff --git a/falcon/util/mediatypes.py b/falcon/util/mediatypes.py index eebed0446..1d5218d5c 100644 --- a/falcon/util/mediatypes.py +++ b/falcon/util/mediatypes.py @@ -16,9 +16,14 @@ from __future__ import annotations -from typing import Dict, Iterator, Tuple +import dataclasses +import functools +import math +from typing import Dict, Iterable, Iterator, Tuple -__all__ = ('parse_header',) +from falcon import errors + +__all__ = ('best_match', 'parse_header', 'quality') def _parse_param_old_stdlib(s: str) -> Iterator[str]: @@ -88,3 +93,231 @@ def parse_header(line: str) -> Tuple[str, Dict[str, str]]: return (key.strip(), pdict) return _parse_header_old_stdlib(line) + + +def _parse_media_type_header(media_type: str) -> Tuple[str, str, dict]: + full_type, params = parse_header(media_type) + + # TODO(vytas): Workaround from python-mimeparse by J. Gregorio et al. + # Do we still need this in 2024? + # Java URLConnection class sends an Accept header that includes a + # single '*'. Turn it into a legal wildcard. + if full_type == '*': + full_type = '*/*' + + main_type, separator, subtype = full_type.partition('/') + if not separator: + raise errors.InvalidMediaType('The media type value must contain type/subtype.') + + return (main_type.strip(), subtype.strip(), params) + + +# TODO(vytas): Should we make these data structures public? + + +# PERF(vytas): It would be nice to use frozen=True as we never modify the data, +# but it seems to incur a performance hit (~2-3x) on CPython 3.12. +@dataclasses.dataclass +class _MediaType: + main_type: str + subtype: str + params: dict + + # NOTE(vytas): Using __slots__ with dataclasses is tricky, but it seems to + # work here since we are not using any default values in the definition. + __slots__ = ('main_type', 'subtype', 'params') + + @classmethod + def parse(cls, media_type: str) -> _MediaType: + return cls(*_parse_media_type_header(media_type)) + + +@dataclasses.dataclass +class _MediaRange: + main_type: str + subtype: str + quality: float + params: dict + + __slots__ = ('main_type', 'subtype', 'quality', 'params') + + _NOT_MATCHING = (-1, -1, -1, -1, 0.0) + + _Q_VALUE_ERROR_MESSAGE = ( + 'If provided, the q parameter must be a real number ' + 'in the range 0 through 1.' + ) + + @classmethod + def parse(cls, media_range: str) -> _MediaRange: + try: + main_type, subtype, params = _parse_media_type_header(media_range) + except errors.InvalidMediaType as ex: + raise errors.InvalidMediaRange( + 'The media range value must contain type/subtype.' + ) from ex + + # NOTE(vytas): We don't need to special-case Q since the above + # parse_header always lowercases parameter names. + + # PERF(vytas): Short-circuit if q is absent. + if 'q' not in params: + return cls(main_type, subtype, 1.0, params) + + try: + q = float(params.pop('q')) + except (TypeError, ValueError) as ex: + # NOTE(vytas): RFC 9110, Section 12.4.2: + # weight = OWS ";" OWS "q=" qvalue + # qvalue = ( "0" [ "." 0*3DIGIT ] ) / ( "1" [ "." 0*3("0") ] ) + raise errors.InvalidMediaRange(cls._Q_VALUE_ERROR_MESSAGE) from ex + + if not (0.0 <= q <= 1.0) or not math.isfinite(q): + raise errors.InvalidMediaRange(cls._Q_VALUE_ERROR_MESSAGE) + + # NOTE(vytas): RFC 9110, Section 12.4.2 states that a sender of qvalue + # MUST NOT generate more than three digits after the decimal point, + # but we are more permissive here, and opt not to spend any extra CPU + # cycles, if we have already managed to convert the value to float. + + return cls(main_type, subtype, q, params) + + def match_score(self, media_type: _MediaType) -> Tuple[int, int, int, int, float]: + if self.main_type == '*' or media_type.main_type == '*': + main_matches = 0 + elif self.main_type != media_type.main_type: + return self._NOT_MATCHING + else: + main_matches = 1 + + if self.subtype == '*' or media_type.subtype == '*': + sub_matches = 0 + elif self.subtype != media_type.subtype: + return self._NOT_MATCHING + else: + sub_matches = 1 + + # PERF(vytas): We could also use bitwise operators directly between + # params.keys(), but set()/frozenset() seem to outperform dict.keys() + # slightly regardless of the number of elements, especially when we + # reuse the same sets for both intersection and symmetric_difference. + mr_pnames = frozenset(self.params) + mt_pnames = frozenset(media_type.params) + + exact_match = 0 if mr_pnames ^ mt_pnames else 1 + + matching = mr_pnames & mt_pnames + for pname in matching: + if self.params[pname] != media_type.params[pname]: + return self._NOT_MATCHING + + return (main_matches, sub_matches, exact_match, len(matching), self.quality) + + +# PERF(vytas): It is possible to cache a classmethod too, but the invocation is +# less efficient, especially in the case of a cache hit. +# NOTE(vytas): Also, if we decide to make these classes public, we need to keep +# these cached parsers private. +_parse_media_type = functools.lru_cache(_MediaType.parse) +_parse_media_range = functools.lru_cache(_MediaRange.parse) + + +@functools.lru_cache() +def _parse_media_ranges(header: str) -> Tuple[_MediaRange, ...]: + return tuple(_MediaRange.parse(media_range) for media_range in header.split(',')) + + +@functools.lru_cache() +def quality(media_type: str, header: str) -> float: + """Get quality of the most specific matching media range. + + Media-ranges are parsed from the provided `header` value according to + RFC 9110, Section 12.5.1 (the ``Accept`` header). + + The provided `media_type` is matched against each of the parsed media + ranges, and the fitness of each match is assessed as follows + (in the decreasing priority list of criteria): + + 1. Do the main types (as in ``type/subtype``) match? + + The types must either match exactly, or as wildcard (``*``). + The matches involving a wildcard are prioritized lower. + + 2. Do the subtypes (as in ``type/subtype``) match? + + The subtypes must either match exactly, or as wildcard (``*``). + The matches involving a wildcard are prioritized lower. + + 3. Do the parameters match exactly? + + If all the parameter names and values (if any) between the media range + and media type match exactly, such match is prioritized higher than + matches involving extraneous parameters on either side. + + Note that if parameter names match, the corresponding values must also + be equal, or the provided media type is considered not to match the + media range in question at all. + + 4. The number of matching parameters. + + 5. Finally, if two or more best media range matches are equally fit + according to all of the above criteria (1) through (4), the highest + quality (i.e., the value of the ``q`` parameter) of these is returned. + + Note: + With the exception of evaluating the exact parameter match (3), the + number of extraneous parameters (i.e. where the names are only present + in the media type, or only in the media range) currently does not + influence the described specificity sort order. + + Args: + media_type: The Internet media type to match against the provided + HTTP ``Accept`` header value. + header: The value of a header that conforms to the format of the + HTTP ``Accept`` header. + + Returns: + Quality of the most specific media range matching the provided + `media_type`. (If none matches, 0.0 is returned.) + """ + parsed_media_type = _parse_media_type(media_type) + most_specific = max( + media_range.match_score(parsed_media_type) + for media_range in _parse_media_ranges(header) + ) + return most_specific[-1] + + +def best_match(media_types: Iterable[str], header: str) -> str: + """Choose media type with the highest :func:`quality` from a list of candidates. + + Args: + media_types: An iterable over one or more Internet media types + to match against the provided header value. + header: The value of a header that conforms to the format of the + HTTP ``Accept`` header. + + Returns: + Best match from the supported candidates, or an empty string if the + provided header value does not match any of the given types. + """ + # PERF(vytas): Using the default parameter, i.e., max(..., default='', 0.0) + # would be much nicer than EAFP, but for some reason it is quite slow + # regardless of whether media_types is empty or not. + try: + matching, best_quality = max( + ((media_type, quality(media_type, header)) for media_type in media_types), + key=lambda mt_quality: mt_quality[1], + ) + if best_quality > 0.0: + return matching + except errors.InvalidMediaType: + # NOTE(vytas): Do not swallow instances of InvalidMediaType + # (it a subclass of ValueError). + raise + except ValueError: + # NOTE(vytas): Barring unknown bugs, we only expect unhandled + # ValueErrors from supplying an empty media_types value. + pass + + return '' diff --git a/falcon/vendor/__init__.py b/falcon/vendor/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/falcon/vendor/mimeparse/LICENSE b/falcon/vendor/mimeparse/LICENSE deleted file mode 100755 index 89de35479..000000000 --- a/falcon/vendor/mimeparse/LICENSE +++ /dev/null @@ -1,17 +0,0 @@ -Permission is hereby granted, free of charge, to any person obtaining a copy -of this software and associated documentation files (the "Software"), to deal -in the Software without restriction, including without limitation the rights -to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -copies of the Software, and to permit persons to whom the Software is -furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in -all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN -THE SOFTWARE. diff --git a/falcon/vendor/mimeparse/__init__.py b/falcon/vendor/mimeparse/__init__.py deleted file mode 100644 index 223767a7b..000000000 --- a/falcon/vendor/mimeparse/__init__.py +++ /dev/null @@ -1,11 +0,0 @@ -""" -This module wraps code from the MIT-licensed python-mimeparse project. - -The original project source code may be obtained from GitHub by visiting the -following URL: - - https://github.com/dbtsai/python-mimeparse - -""" - -from .mimeparse import * # NOQA diff --git a/falcon/vendor/mimeparse/mimeparse.py b/falcon/vendor/mimeparse/mimeparse.py deleted file mode 100755 index f96e63384..000000000 --- a/falcon/vendor/mimeparse/mimeparse.py +++ /dev/null @@ -1,191 +0,0 @@ -from falcon.util.mediatypes import parse_header - -__version__ = '1.6.0' -__author__ = 'Joe Gregorio' -__email__ = 'joe@bitworking.org' -__license__ = 'MIT License' -__credits__ = '' - - -class MimeTypeParseException(ValueError): - pass - - -def parse_mime_type(mime_type): - """Parses a mime-type into its component parts. - - Carves up a mime-type and returns a tuple of the (type, subtype, params) - where 'params' is a dictionary of all the parameters for the media range. - For example, the media range 'application/xhtml;q=0.5' would get parsed - into: - - ('application', 'xhtml', {'q', '0.5'}) - - :rtype: (str,str,dict) - """ - full_type, params = parse_header(mime_type) - # Java URLConnection class sends an Accept header that includes a - # single '*'. Turn it into a legal wildcard. - if full_type == '*': - full_type = '*/*' - - type_parts = full_type.split('/') if '/' in full_type else None - if not type_parts or len(type_parts) > 2: - raise MimeTypeParseException( - "Can't parse type \"{}\"".format(full_type)) - - (type, subtype) = type_parts - - return (type.strip(), subtype.strip(), params) - - -def parse_media_range(range): - """Parse a media-range into its component parts. - - Carves up a media range and returns a tuple of the (type, subtype, - params) where 'params' is a dictionary of all the parameters for the media - range. For example, the media range 'application/*;q=0.5' would get parsed - into: - - ('application', '*', {'q', '0.5'}) - - In addition this function also guarantees that there is a value for 'q' - in the params dictionary, filling it in with a proper default if - necessary. - - :rtype: (str,str,dict) - """ - (type, subtype, params) = parse_mime_type(range) - params.setdefault('q', params.pop('Q', None)) # q is case insensitive - try: - if not params['q'] or not 0 <= float(params['q']) <= 1: - params['q'] = '1' - except ValueError: # from float() - params['q'] = '1' - - return (type, subtype, params) - - -def quality_and_fitness_parsed(mime_type, parsed_ranges): - """Find the best match for a mime-type amongst parsed media-ranges. - - Find the best match for a given mime-type against a list of media_ranges - that have already been parsed by parse_media_range(). Returns a tuple of - the fitness value and the value of the 'q' quality parameter of the best - match, or (-1, 0) if no match was found. Just as for quality_parsed(), - 'parsed_ranges' must be a list of parsed media ranges. - - :rtype: (float,int) - """ - best_fitness = -1 - best_fit_q = 0 - (target_type, target_subtype, target_params) = \ - parse_media_range(mime_type) - - for (type, subtype, params) in parsed_ranges: - - # check if the type and the subtype match - type_match = ( - type in (target_type, '*') or - target_type == '*' - ) - subtype_match = ( - subtype in (target_subtype, '*') or - target_subtype == '*' - ) - - # if they do, assess the "fitness" of this mime_type - if type_match and subtype_match: - - # 100 points if the type matches w/o a wildcard - fitness = type == target_type and 100 or 0 - - # 10 points if the subtype matches w/o a wildcard - fitness += subtype == target_subtype and 10 or 0 - - # 1 bonus point for each matching param besides "q" - param_matches = sum([ - 1 for (key, value) in target_params.items() - if key != 'q' and key in params and value == params[key] - ]) - fitness += param_matches - - # finally, add the target's "q" param (between 0 and 1) - fitness += float(target_params.get('q', 1)) - - if fitness > best_fitness: - best_fitness = fitness - best_fit_q = params['q'] - - return float(best_fit_q), best_fitness - - -def quality_parsed(mime_type, parsed_ranges): - """Find the best match for a mime-type amongst parsed media-ranges. - - Find the best match for a given mime-type against a list of media_ranges - that have already been parsed by parse_media_range(). Returns the 'q' - quality parameter of the best match, 0 if no match was found. This function - behaves the same as quality() except that 'parsed_ranges' must be a list of - parsed media ranges. - - :rtype: float - """ - - return quality_and_fitness_parsed(mime_type, parsed_ranges)[0] - - -def quality(mime_type, ranges): - """Return the quality ('q') of a mime-type against a list of media-ranges. - - Returns the quality 'q' of a mime-type when compared against the - media-ranges in ranges. For example: - - >>> quality('text/html','text/*;q=0.3, text/html;q=0.7, - text/html;level=1, text/html;level=2;q=0.4, */*;q=0.5') - 0.7 - - :rtype: float - """ - parsed_ranges = [parse_media_range(r) for r in ranges.split(',')] - - return quality_parsed(mime_type, parsed_ranges) - - -def best_match(supported, header): - """Return mime-type with the highest quality ('q') from list of candidates. - - Takes a list of supported mime-types and finds the best match for all the - media-ranges listed in header. The value of header must be a string that - conforms to the format of the HTTP Accept: header. The value of 'supported' - is a list of mime-types. The list of supported mime-types should be sorted - in order of increasing desirability, in case of a situation where there is - a tie. - - >>> best_match(['application/xbel+xml', 'text/xml'], - 'text/*;q=0.5,*/*; q=0.1') - 'text/xml' - - :rtype: str - """ - split_header = _filter_blank(header.split(',')) - parsed_header = [parse_media_range(r) for r in split_header] - weighted_matches = [] - pos = 0 - for mime_type in supported: - weighted_matches.append(( - quality_and_fitness_parsed(mime_type, parsed_header), - pos, - mime_type - )) - pos += 1 - weighted_matches.sort() - - return weighted_matches[-1][0][0] and weighted_matches[-1][2] or '' - - -def _filter_blank(i): - """Return all non-empty items in the list.""" - for s in i: - if s.strip(): - yield s diff --git a/pyproject.toml b/pyproject.toml index ebccd4505..4fe30640a 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -84,7 +84,6 @@ exclude = ["examples", "tests"] exclude = [ "falcon/bench", "falcon/cmd", - "falcon/vendor", ] disallow_untyped_defs = true warn_unused_ignores = true @@ -114,12 +113,6 @@ exclude = ["examples", "tests"] ] ignore_missing_imports = true - [[tool.mypy.overrides]] - module = [ - "falcon.vendor.*", - ] - disallow_untyped_defs = false - [tool.towncrier] package = "falcon" package_dir = "" @@ -159,7 +152,6 @@ exclude = ["examples", "tests"] target-version = ["py38"] skip-string-normalization = true line-length = 88 - extend-exclude = "falcon/vendor" [tool.blue] # NOTE(vytas): Before switching to Ruff, Falcon used the Blue formatter. @@ -167,13 +159,11 @@ exclude = ["examples", "tests"] # only minor cosmetic changes in a handful of files. target-version = ["py38"] line-length = 88 - extend-exclude = "falcon/vendor" [tool.ruff] target-version = "py38" format.quote-style = "single" line-length = 88 - extend-exclude = ["falcon/vendor"] builtins = [ "ignore", "attr", diff --git a/setup.py b/setup.py index 9751f825c..3e5ce1e99 100644 --- a/setup.py +++ b/setup.py @@ -41,7 +41,6 @@ def list_modules(dirname, pattern): 'falcon.media', 'falcon.routing', 'falcon.util', - 'falcon.vendor.mimeparse', ] modules_to_exclude = [ @@ -59,6 +58,7 @@ def list_modules(dirname, pattern): # NOTE(vytas): It is pointless to cythonize reader.py, since cythonized # Falcon is using reader.pyx instead. 'falcon.hooks', + 'falcon.inspect', 'falcon.responders', 'falcon.util.reader', 'falcon.util.sync', diff --git a/tests/test_examples.py b/tests/test_examples.py index 5fd5b1db7..e203b8233 100644 --- a/tests/test_examples.py +++ b/tests/test_examples.py @@ -13,6 +13,18 @@ import falcon.testing as testing +def test_quote(util): + quote = util.load_module('examples/quote.py') + + resp = testing.simulate_get(quote.app, '/quote') + + assert resp.status_code == 200 + assert resp.json == { + 'author': 'Grace Hopper', + 'quote': "I've always been more interested in the future than in the past.", + } + + def test_things(asgi, util): suffix = '_asgi' if asgi else '' things = util.load_module(f'examples/things{suffix}.py') diff --git a/tests/test_mediatypes.py b/tests/test_mediatypes.py index 0fae79b43..7093a5175 100644 --- a/tests/test_mediatypes.py +++ b/tests/test_mediatypes.py @@ -1,5 +1,8 @@ +import itertools + import pytest +from falcon import errors from falcon.util import mediatypes @@ -39,3 +42,242 @@ ) def test_parse_header(value, expected): assert mediatypes.parse_header(value) == expected + + +def test_media_type_private_cls(): + mt1 = mediatypes._MediaType.parse('image/png') + assert mt1.main_type == 'image' + assert mt1.subtype == 'png' + assert mt1.params == {} + + mt2 = mediatypes._MediaType.parse('text/plain; charset=latin-1') + assert mt2.main_type == 'text' + assert mt2.subtype == 'plain' + assert mt2.params == {'charset': 'latin-1'} + + +def test_media_range_private_cls(): + mr1 = mediatypes._MediaRange.parse('image/png') + assert mr1.main_type == 'image' + assert mr1.subtype == 'png' + assert mr1.quality == 1.0 + assert mr1.params == {} + + mr2 = mediatypes._MediaRange.parse('text/plain; charset=latin-1; Q=0.9') + assert mr2.main_type == 'text' + assert mr2.subtype == 'plain' + assert pytest.approx(mr2.quality) == 0.9 + assert mr2.params == {'charset': 'latin-1'} + + mr3 = mediatypes._MediaRange.parse('*; q=0.7') + assert mr3.main_type == '*' + assert mr3.subtype == '*' + assert pytest.approx(mr3.quality) == 0.7 + assert mr3.params == {} + + +_RFC_7231_EXAMPLE_ACCEPT = ( + 'text/*;q=0.3, text/html;q=0.7, text/html;level=1, ' + 'text/html;level=2;q=0.4, */*;q=0.5' +) +_RFC_7231_EXAMPLE_VALUES = [ + ('text/html;level=1', 1.0), + ('text/html', 0.7), + ('text/plain', 0.3), + ('image/jpeg', 0.5), + ('text/html;level=2', 0.4), + ('text/html;level=3', 0.7), +] + +_RFC_9110_EXAMPLE_ACCEPT = ( + 'text/*;q=0.3, text/plain;q=0.7, text/plain;format=flowed, ' + 'text/plain;format=fixed;q=0.4, */*;q=0.5' +) +# NOTE(vytas): Including the errata https://www.rfc-editor.org/errata/eid7138. +_RFC_9110_EXAMPLE_VALUES = [ + ('text/plain;format=flowed', 1.0), + ('text/plain', 0.7), + ('text/html', 0.3), + ('image/jpeg', 0.5), + ('text/plain;format=fixed', 0.4), + ('text/html;level=3', 0.3), +] + +_RFC_EXAMPLES = list( + itertools.chain.from_iterable( + ((accept,) + media_type_quality for media_type_quality in example_values) + for accept, example_values in ( + (_RFC_7231_EXAMPLE_ACCEPT, _RFC_7231_EXAMPLE_VALUES), + (_RFC_9110_EXAMPLE_ACCEPT, _RFC_9110_EXAMPLE_VALUES), + ) + ) +) + +_RFC_EXAMPLE_IDS = list( + itertools.chain.from_iterable( + ( + f'{rfc}-{media_type}-{quality_value}' + for media_type, quality_value in example_values + ) + for rfc, example_values in ( + ('RFC-7231', _RFC_7231_EXAMPLE_VALUES), + ('RFC-9110', _RFC_7231_EXAMPLE_VALUES), + ) + ) +) + + +@pytest.mark.parametrize( + 'accept,media_type,quality_value', _RFC_EXAMPLES, ids=_RFC_EXAMPLE_IDS +) +def test_quality_rfc_examples(accept, media_type, quality_value): + assert pytest.approx(mediatypes.quality(media_type, accept)) == quality_value + + +@pytest.mark.parametrize( + 'accept,media_type,quality_value', + [ + ( + 'application/*, */wildcard; q=0.7, */*; q=0.25', + 'test/wildcard; expect=pass', + 0.7, + ), + ( + 'application/*, */wildcard; q=0.7, */*; q=0.25', + 'application/wildcard; expect=pass', + 1.0, + ), + ( + 'application/*, */wildcard; q=0.7, */*; q=0.25', + 'test/something; expect=pass', + 0.25, + ), + ( + 'text/x-python, text/*; q=0.33, text/plain; format=fixed', + 'text/plain; format=flowed', + 0.33, + ), + ( + # NOTE(vytas): Same as one of the RFC 7231 examples, just with some + # media ranges reordered. python-mimeparse fails to yield the + # correct result in this specific case. + 'text/*;q=0.3, text/html;level=1, text/html;q=0.7, ' + 'text/html;level=2;q=0.4, */*;q=0.5', + 'text/html; level=3', + 0.7, + ), + ], +) +def test_quality(accept, media_type, quality_value): + assert pytest.approx(mediatypes.quality(media_type, accept)) == quality_value + + +@pytest.mark.parametrize( + 'accept,media_type', + [ + ( + 'foo/bar, test/app; q=0.2, test/app; p=1; q=0.9, test/app;p=1;r=2', + 'test/app', + ), + ('test/app; q=0.1, test/app; p=1; q=0.2, test/app;p=1;r=2', 'test/app; p=1'), + ( + '*/app; q=0.1, simple/app; test=true; q=0.2, simple/app; color=blue', + 'simple/app; test=true', + ), + ], +) +def test_quality_prefer_exact_match(accept, media_type): + assert pytest.approx(mediatypes.quality(media_type, accept)) == 0.2 + + +_QUALITY_NONE_MATCHES_EXAMPLES = [ + ('application/json', 'application/yaml'), + ('audio/*; q=0.2, audio/basic', 'video/mp3'), + ( + 'falcon/peregrine; speed=high; unladen=true', + 'falcon/peregrine; speed=average', + ), + ('text/html, text/plain', 'text/x-python'), + ('*/json; q=0.2, application/json', 'application/msgpack'), + ( + 'text/x-python, image/*; q=0.33, text/plain; format=fixed', + 'text/plain; format=flowed', + ), +] + + +@pytest.mark.parametrize('accept,media_type', _QUALITY_NONE_MATCHES_EXAMPLES) +def test_quality_none_matches(accept, media_type): + assert mediatypes.quality(media_type, accept) == 0.0 + + +@pytest.mark.parametrize( + 'media_types,accept,expected', + [ + (['application/json'], 'application/json', 'application/json'), + (['application/json'], 'application/json; charset=utf-8', 'application/json'), + ( + ['application/json', 'application/yaml'], + 'application/json, */*; q=0.2', + 'application/json', + ), + ], +) +def test_best_match(media_types, accept, expected): + assert mediatypes.best_match(media_types, accept) == expected + + +_BEST_MATCH_NONE_MATCHES_EXAMPLES = [ + ([_mt], _acc) for _mt, _acc in _QUALITY_NONE_MATCHES_EXAMPLES +] + [ + (['application/json', 'application/yaml'], 'application/xml, text/*; q=0.7'), + ( + ['text/plain', 'falcon/peregrine; load=unladen'], + 'falcon/peregrine; load=heavy', + ), +] + + +@pytest.mark.parametrize('media_types,accept', _BEST_MATCH_NONE_MATCHES_EXAMPLES) +def test_best_match_none_matches(media_types, accept): + assert mediatypes.best_match(media_types, accept) == '' + + +@pytest.mark.parametrize('media_type', ['', 'word document', 'text']) +def test_invalid_media_type(media_type): + with pytest.raises(errors.InvalidMediaType): + mediatypes.quality(media_type, '*/*') + + +def _generate_strings(items): + yield from items + + +@pytest.mark.parametrize( + 'media_range', + [ + '', + 'word document', + 'text', + 'text/plain; q=high', + '*/*; q=inf', + '*/*; q=-inf', + '*/*; q=nan', + 'application/very-important; q=1337.0', + ], +) +def test_invalid_media_range(media_range): + with pytest.raises(errors.InvalidMediaRange): + mediatypes.quality('falcon/peregrine', media_range) + + with pytest.raises(errors.InvalidMediaRange): + mediatypes.best_match(_generate_strings(['falcon/peregrine']), media_range) + + +@pytest.mark.parametrize( + 'accept', + ['*/*', 'application/xml, text/*; q=0.7, */*; q=0.3'], +) +@pytest.mark.parametrize('media_types', [(), [], {}, _generate_strings(())]) +def test_empty_media_types(accept, media_types): + assert mediatypes.best_match(media_types, accept) == '' diff --git a/tools/testing/install_hug.sh b/tools/testing/install_hug.sh deleted file mode 100755 index 67e1ab8eb..000000000 --- a/tools/testing/install_hug.sh +++ /dev/null @@ -1,23 +0,0 @@ -#!/usr/bin/env bash - -HUG_PATH=.ecosystem/hug - -# Clean up in case we are running locally and not in CI -rm -rf $HUG_PATH - -mkdir -p .ecosystem -git clone https://github.com/timothycrosley/hug.git $HUG_PATH -virtualenv $HUG_PATH/.venv -source $HUG_PATH/.venv/bin/activate - -pip install hug - -pushd $HUG_PATH -git checkout master -git pull -HUG_VERSION=$(pip freeze | grep hug | cut -c 6-) -git checkout tags/$HUG_VERSION -pip install -rrequirements/build.txt -popd - -pip install . # Override Hug's Falcon version with the one under test diff --git a/tools/testing/test_hug.sh b/tools/testing/test_hug.sh deleted file mode 100755 index 85d656e5a..000000000 --- a/tools/testing/test_hug.sh +++ /dev/null @@ -1,8 +0,0 @@ -#!/usr/bin/env bash - -set -e - -pushd .ecosystem/hug -source .venv/bin/activate -python -m pytest tests -popd diff --git a/tox.ini b/tox.ini index 0b96f9d77..767a018c7 100644 --- a/tox.ini +++ b/tox.ini @@ -15,6 +15,7 @@ envlist = cleanup, ruff, pep8, + pep8-docstrings, mypy, mypy_tests, mintest, @@ -48,29 +49,6 @@ commands = coverage combine coverage html -d {toxinidir}/.coverage_html coverage report --fail-under 100 -# -------------------------------------------------------------------- -# ASGI servers -# -------------------------------------------------------------------- - -[testenv:daphne] -deps = {[testenv]deps} - daphne - -[testenv:hypercorn] -# TODO(vytas): Unpin Hypercorn once the buggy versions are yanked upstream. -deps = {[testenv]deps} - hypercorn != 0.14.0, != 0.14.1 - -# -------------------------------------------------------------------- -# Test without optional packages -# -------------------------------------------------------------------- - -[testenv:no_optional_packages] -deps = {[testenv]deps} -commands = python "{toxinidir}/tools/clean.py" "{toxinidir}/falcon" - pip uninstall --yes python-rapidjson daphne orjson - pytest tests [] - # -------------------------------------------------------------------- # Coverage # -------------------------------------------------------------------- @@ -93,11 +71,6 @@ deps = {[testenv]deps} jsonschema commands = coverage run -m pytest tests [] -[testenv:pytest_sans_msgpack] -deps = {[testenv]deps} -commands = pip uninstall --yes msgpack - coverage run -m pytest tests -k "test_ws and test_msgpack_missing" - [testenv:py312] basepython = python3.12 deps = {[testenv]deps} @@ -105,12 +78,6 @@ deps = {[testenv]deps} jsonschema commands = {[with-coverage]commands} -[testenv:py312_sans_msgpack] -basepython = python3.12 -deps = {[testenv]deps} -commands = pip uninstall --yes msgpack - coverage run -m pytest tests -k "test_ws and test_msgpack_missing" - [testenv:py312_nocover] basepython = python3.12 deps = {[testenv]deps} @@ -134,13 +101,13 @@ deps = -r{toxinidir}/requirements/tests pdbpp [testenv:py3_debug] -basepython = python3.10 +basepython = python3.12 deps = {[with-debug-tools]deps} uvicorn jsonschema # -------------------------------------------------------------------- -# mypy +# Mypy # -------------------------------------------------------------------- [testenv:mypy] @@ -229,12 +196,33 @@ commands = {[with-cython]commands} setenv = {[with-cython]setenv} deps = {[with-cython]deps} gunicorn - # NOTE(caselit): meinheld requires greenlet<0.5 that's not compatible with py3.10 - meinheld uwsgi waitress commands = pytest -v tests/test_wsgi_servers.py +[testenv:wsgi_meinheld] +# NOTE(caselit): meinheld requires greenlet<0.5 that's not compatible with py3.10 +basepython = python3.8 +setenv = {[with-cython]setenv} +deps = {[with-cython]deps} + gunicorn + meinheld +commands = pytest -v tests/test_wsgi_servers.py + +# -------------------------------------------------------------------- +# ASGI servers +# -------------------------------------------------------------------- + +[testenv:daphne] +deps = {[testenv]deps} + daphne +commands = pytest -v tests/asgi/test_asgi_servers.py + +[testenv:hypercorn] +deps = {[testenv]deps} + hypercorn +commands = pytest -v tests/asgi/test_asgi_servers.py + # -------------------------------------------------------------------- # Smoke testing with a sample app # -------------------------------------------------------------------- @@ -266,8 +254,17 @@ commands = {[smoke-test]commands} [testenv:pep8] deps = ruff +skip_install = True commands = ruff check [] +[testenv:pep8-docstrings] +deps = ruff +skip_install = True +commands = ruff check \ + --exclude=.ecosystem,.eggs,.git,.tox,.venv,build,dist,docs,examples,tests,falcon/bench/nuts \ + --select=D205,D212,D400,D401,D403,D404 \ + [] + [testenv:ruff] deps = ruff>=0.3.7 skip_install = True @@ -276,35 +273,23 @@ commands = ruff format --check . [] [testenv:reformat] deps = ruff>=0.3.7 skip_install = True -commands = ruff format . [] - -[testenv:pep8-docstrings] -deps = ruff -commands = ruff check \ - --exclude=.ecosystem,.eggs,.git,.tox,.venv,build,dist,docs,examples,tests,falcon/vendor,falcon/bench/nuts \ - --select=D205,D212,D400,D401,D403,D404 \ - [] - -[testenv:pep8-examples] -deps = ruff -commands = ruff check examples [] +commands = + ruff format . [] + ruff check --fix # -------------------------------------------------------------------- # For viewing environ dicts generated by various WSGI servers # -------------------------------------------------------------------- -[testenv:py310_dump_gunicorn] -basepython = python3.10 +[testenv:dump_gunicorn] deps = gunicorn commands = gunicorn -b localhost:8000 tests.dump_wsgi -[testenv:py310_dump_waitress] -basepython = python3.10 +[testenv:dump_waitress] deps = waitress commands = waitress-serve --listen=localhost:8000 tests.dump_wsgi:application -[testenv:py310_dump_wsgiref] -basepython = python3.10 +[testenv:dump_wsgiref] commands = python tests/dump_wsgi.py # -------------------------------------------------------------------- @@ -343,28 +328,16 @@ basepython = pypy3 deps = -r{toxinidir}/requirements/bench commands = falcon-bench [] -# -------------------------------------------------------------------- -# Check for new versions of vendored packages -# -------------------------------------------------------------------- - -[testenv:check_vendored] -basepython = python3.10 -allowlist_externals = {toxinidir}/tools/check-vendored.sh -deps = -commands = {toxinidir}/tools/check-vendored.sh - # -------------------------------------------------------------------- # Package sanity check with twine # -------------------------------------------------------------------- [testenv:twine_check] -basepython = python3.10 skipsdist = True -deps = setuptools +deps = build twine - wheel commands = - python {toxinidir}/setup.py bdist_wheel sdist + python -m build twine check {toxinidir}/dist/* # -------------------------------------------------------------------- @@ -414,7 +387,11 @@ commands = {toxinidir}/tools/generate_dash.sh # -------------------------------------------------------------------- -# Tutorial ("look") tests +# Tutorial tests +# +# - Tutorial ("look") tests +# - ASGI tutorial ("asgilook") tests +# - WebSockets tutorial ("ws_tutorial") tests # -------------------------------------------------------------------- [testenv:look] @@ -423,10 +400,6 @@ deps = commands = pytest {toxinidir}/examples/look/tests -# -------------------------------------------------------------------- -# ASGI tutorial ("asgilook") tests -# -------------------------------------------------------------------- - [testenv:asgilook] basepython = python3.12 deps = @@ -440,10 +413,6 @@ commands = --cov-report term-missing \ {toxinidir}/examples/asgilook/tests/ -# -------------------------------------------------------------------- -# WebSockets tutorial ("ws_tutorial") tests -# -------------------------------------------------------------------- - [testenv:ws_tutorial] basepython = python3.12 deps = @@ -457,17 +426,6 @@ commands = --cov-report term-missing \ {toxinidir}/examples/ws_tutorial/tests/ -# -------------------------------------------------------------------- -# Ecosystem -# -------------------------------------------------------------------- - -[testenv:hug] -basepython = python3.8 -deps = virtualenv -commands = - {toxinidir}/tools/testing/install_hug.sh - {toxinidir}/tools/testing/test_hug.sh - # -------------------------------------------------------------------- # E2E tests # -------------------------------------------------------------------- @@ -485,4 +443,3 @@ deps = -r{toxinidir}/requirements/e2e commands = pytest {toxinidir}/e2e-tests/ --browser=firefox -