doc: patch Sphinx to detect our `@final` for marking classes as `final`

Thanks to Dominic Davis-Foster for code & assistance.
This commit is contained in:
Ran Benita 2020-10-03 13:10:03 +03:00
parent a6a7ba57e0
commit 7705e5e624
2 changed files with 22 additions and 0 deletions

1
changelog/7780.doc.rst Normal file
View File

@ -0,0 +1 @@
Classes which should not be inherited from are now marked ``final class`` in the API reference.

View File

@ -15,8 +15,10 @@
# #
# The full version, including alpha/beta/rc tags. # The full version, including alpha/beta/rc tags.
# The short X.Y version. # The short X.Y version.
import ast
import os import os
import sys import sys
from typing import List
from typing import TYPE_CHECKING from typing import TYPE_CHECKING
from _pytest import __version__ as version from _pytest import __version__ as version
@ -398,3 +400,22 @@ def setup(app: "sphinx.application.Sphinx") -> None:
) )
configure_logging(app) configure_logging(app)
# Make Sphinx mark classes with "final" when decorated with @final.
# We need this because we import final from pytest._compat, not from
# typing (for Python < 3.8 compat), so Sphinx doesn't detect it.
# To keep things simple we accept any `@final` decorator.
# Ref: https://github.com/pytest-dev/pytest/pull/7780
import sphinx.pycode.ast
import sphinx.pycode.parser
original_is_final = sphinx.pycode.parser.VariableCommentPicker.is_final
def patched_is_final(self, decorators: List[ast.expr]) -> bool:
if original_is_final(self, decorators):
return True
return any(
sphinx.pycode.ast.unparse(decorator) == "final" for decorator in decorators
)
sphinx.pycode.parser.VariableCommentPicker.is_final = patched_is_final