Skip to content

Commit

Permalink
no-member: Support tracking/ignoring request.cls.X / self.X
Browse files Browse the repository at this point in the history
It seems we already support the (weird) scenario that someone does
`clls = request.cls`, and subsequently uses `clls.X = Y`

However, it appears that we are missing the trivial `request.cls.X = Y` case

Fixes #74

Additionally:

* Factorize condititons for semantic naming
* Make some "not-docstrings" comments, and fix some grammar issues.

In order to "properly" analyze fixtures that might change the class itself
(which it can usually be out of order typed), we need to walk the
`fixture` `FunctionDef`s in advance, in their topological order, per-scope.

For that, we need to hand-roll out own ASTWalker / ASTVisitor
(pylint's is a bit tight to their `checker` implementation).

This enables us to (almost) correctly allow pylint to introspect
`used-before-assignment` issues
(see `tests/input/no-member/not_using_cls.py` changes)

For project internals:

* Split `pylint_pytest/utils.py` to modules

Signed-off-by: Stavros Ntentos <133706+stdedos@users.noreply.github.com>
  • Loading branch information
stdedos committed Aug 25, 2024
1 parent e3deae9 commit 2066591
Show file tree
Hide file tree
Showing 14 changed files with 537 additions and 82 deletions.
2 changes: 2 additions & 0 deletions .pre-commit-config.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -79,7 +79,9 @@ repos:
- id: mypy
exclude: tests/input/
additional_dependencies:
- attrs
- pylint
- types-networkx
- repo: local
hooks:
- id: pylint
Expand Down
2 changes: 1 addition & 1 deletion pylint_pytest/checkers/__init__.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
from pylint.checkers import BaseChecker

from pylint_pytest.utils import PYLINT_VERSION_MAJOR
from pylint_pytest.utils.pytest_logic import PYLINT_VERSION_MAJOR


class BasePytestChecker(BaseChecker):
Expand Down
268 changes: 226 additions & 42 deletions pylint_pytest/checkers/class_attr_loader.py
Original file line number Diff line number Diff line change
@@ -1,56 +1,240 @@
from __future__ import annotations

from astroid import Assign, Attribute, ClassDef, Name
from typing import cast

from ..utils import _can_use_fixture, _is_class_autouse_fixture
from . import BasePytestChecker
from astroid import Assign, AssignAttr, Attribute, ClassDef, FunctionDef, Name, NodeNG

from pylint_pytest.checkers import BasePytestChecker
from pylint_pytest.utils.ast_walker import ASTVisitor, ASTWalker
from pylint_pytest.utils.pytest_logic import (
AssignPair,
FixtureProperties,
FixtureScopes,
FixturesStructure,
_get_fixture_kwarg,
_is_fixture_function,
_is_test_function,
fixtures_and_deps_in_topo_order,
)

CLASS_CONVENTIONAL_NAME = "cls"
REQUEST_FIXTURE_NAME = "request"


def _is_request_cls(attr_node: NodeNG) -> bool:
"""Is node exactly ``request.cls.*``?"""

return (
isinstance(attr_node, Attribute)
and attr_node.attrname == CLASS_CONVENTIONAL_NAME
and isinstance(attr_node.expr, Name)
and attr_node.expr.name == REQUEST_FIXTURE_NAME
)


class ClassAttrLoader(BasePytestChecker):
"""
This checker is responsible for pre-processing any class-defined fixtures,
and ensure that test classes' ``self`` see the attributes defined in the fixtures.
The algorithm is as follows:
#. When visiting a testing file, from top to bottom
#. When visiting a testing class
#. Collect all fixtures, their properties
#. Calculate the dependencies between fixtures
#. Evaluate auto fixtures in scope-order.
#. Use ``nx.DiGraph`` to topologically sort the fixtures.
#. Evaluate the fixtures in order
#. Use a thin ``FixtureVisitor`` to evaluate the fixtures.
#. Resume the class visitation
#. When visiting a fixture or a test function
#. Evaluate any reference-by-name fixture requested.
#. For fixtures:
#. Mark a fixture as AST-evaluated.
#. Apply any hacks, as-visited
#. When leaving, revert any hacks applied in this scope.
"""

msgs = {"E6400": ("", "pytest-class-attr-loader", "")}

in_setup = False
request_cls: set[str] = set()
class_node: ClassDef | None = None
fixtures: FixturesStructure = {}

curr_fixture: FunctionDef | None = None
request_cls_aliases: set[str] = set()

ooo_applied_assigns: set[AssignPair] = set()

def visit_classdef(self, node: ClassDef):
if not (node.name.lower().startswith("test") or "unittest.TestCase" in node.basenames):
return

self.class_node = node

for fn_node in node.nodes_of_class(FunctionDef):
if not _is_fixture_function(fn_node):
continue

def visit_functiondef(self, node):
"""determine if a method is a class setup method"""
self.in_setup = False
self.request_cls = set()
self.fixtures[fn_node] = FixtureProperties(
name=repr(fn_node),
autouse=_get_fixture_kwarg(fn_node, "autouse", False),
scope=FixtureScopes.from_str(
_get_fixture_kwarg(fn_node, "scope", str(FixtureScopes.FUNCTION))
),
)

for fixture_fn, fixture_data in self.fixtures.items():
for param in fixture_fn.args.args:
for other_fn, _ in self.fixtures.items():
if param.name is other_fn.name:
fixture_data.deps.add(other_fn)

self.evaluate_auto_fixtures()

def leave_classdef(self, _node: ClassDef):
self.fixtures.clear()
self.class_node = None

if _can_use_fixture(node) and _is_class_autouse_fixture(node):
self.in_setup = True
self.class_node = node.parent
def evaluate_auto_fixtures(self):
for scope in FixtureScopes:
filtered_fixtures = {f: d for f, d in self.fixtures.items() if d.scope == scope}
order = fixtures_and_deps_in_topo_order(filtered_fixtures)

for fixture_fn in order:
if not self.fixtures[fixture_fn].autouse:
continue

if self.fixtures[fixture_fn].pre_evaluated:
continue

self.walk_fixture(fixture_fn)

def walk_fixture(self, fixture_fn: FunctionDef):
walker = ASTWalker()
walker.add_visitor(FixtureVisitor(fixture_fn, self))
walker.walk(fixture_fn)
self.fixtures[fixture_fn].pre_evaluated = True
self.curr_fixture = None
self.request_cls_aliases.clear()

def __add_to_locals(self, dest_node: AssignAttr, value_node: Assign):
self.class_node = cast(ClassDef, self.class_node) # XXX: Cannot fight anymore
self.class_node.locals[dest_node.attrname] = [value_node.value]

def visit_functiondef(self, fn_node: FunctionDef):
if not self.class_node or (not _is_test_function(fn_node) and fn_node not in self.fixtures):
return

self.__ooo_apply_fixtures(fn_node)
if _is_test_function(fn_node):
return

if not self.fixtures[fn_node].pre_evaluated:
self.walk_fixture(fn_node)

self.curr_fixture = fn_node
self.fixtures[fn_node].ast_evaluating = True

def __ooo_apply_fixtures(self, fn_node: FunctionDef):
"""
Apply all relevant fixtures to the current function's class' namespace.
"""
sort_fixtures: FixturesStructure = {}

# Detect any referenced fixtures in function arguments
for arg in fn_node.args.args:
for fixture_fn, fixture_data in self.fixtures.items():
if arg.name != fixture_fn.name:
continue

sort_fixtures[fixture_fn] = fixture_data

# Add all autouse fixtures
sort_fixtures.update(
{f: d for f, d in self.fixtures.items() if d.autouse and f is not fn_node}
)

sorted_fixtures = fixtures_and_deps_in_topo_order(sort_fixtures)

for fixture_fn in sorted_fixtures:
fixture_data = self.fixtures[fixture_fn]

if not fixture_data.pre_evaluated:
# Not pre-evaluated, so that means
# we have no ``fixture_data.assign_attrs``.
self.walk_fixture(fixture_fn)

for pair in fixture_data.assign_attrs:
self.__add_to_locals(pair.attr, pair.assign)
self.ooo_applied_assigns.add(pair)

def leave_functiondef(self, fn_node: FunctionDef):
if not self.class_node:
return

for pair in self.ooo_applied_assigns:
del self.class_node.locals[pair.attr.attrname]

self.ooo_applied_assigns.clear()

if fn_node in self.fixtures:
self.curr_fixture = None
self.request_cls_aliases.clear()

def visit_assign(self, node: Assign):
"""store the aliases for `cls`"""
if (
self.in_setup
and isinstance(node.value, Attribute)
and node.value.attrname == "cls"
and isinstance(node.value.expr, Name)
and node.value.expr.name == "request"
):
# storing the aliases for cls from request.cls
self.request_cls = set(t.name for t in node.targets)

def visit_assignattr(self, node):
if (
self.in_setup
and isinstance(node.expr, Name)
and node.expr.name in self.request_cls
and self.class_node is not None
and node.attrname not in self.class_node.locals
):
try:
# find Assign node which contains the source "value"
assign_node = node
while not isinstance(assign_node, Assign):
assign_node = assign_node.parent

# hack class locals
self.class_node.locals[node.attrname] = [assign_node.value]
except Exception: # pylint: disable=broad-except
# cannot find valid assign expr, skipping the entire attribute
pass
if not self.curr_fixture or not _is_request_cls(node.value):
return

self.request_cls_aliases = set(t.name for t in node.targets)

def visit_assignattr(self, node: AssignAttr):
# "Intra"-apply any `request.cls.X` assign "also" to `self.X`.
# "External" apply is handled in ``self.__ooo_apply_fixtures``.
self.class_node = cast(ClassDef, self.class_node) # XXX: Cannot fight anymore
if not self.eligible_for_assignattr_visit(node) or node.attrname in self.class_node.locals:
return

# We did all the heavy lifting in the FixtureVisitor; now, to hack:
for pair in self.fixtures[self.curr_fixture].assign_attrs:
if pair.attr is node:
self.__add_to_locals(pair.attr, pair.assign)
return

def eligible_for_assignattr_visit(self, node: AssignAttr) -> bool:
return (
self.class_node is not None
and self.curr_fixture is not None
and (self.is_node_request_cls_alias(node) or _is_request_cls(node.expr))
)

def is_node_request_cls_alias(self, node: AssignAttr) -> bool:
"""Is node a ``var``, where ``var = request.cls``?"""
return isinstance(node.expr, Name) and node.expr.name in self.request_cls_aliases


class FixtureVisitor(ASTVisitor):
def __init__(self, fixture: FunctionDef, checker: ClassAttrLoader):
self.checker = checker
self.checker.curr_fixture = fixture

def visit_assign(self, node: Assign):
self.checker.visit_assign(node)

def visit_assignattr(self, node: AssignAttr):
if not self.checker.eligible_for_assignattr_visit(node):
return

try:
# Find the ``Assign`` node, which contains the source "value"
assign_node = node
while not isinstance(assign_node, Assign):
assign_node = assign_node.parent

# When the time comes, we will hack the class locals
self.checker.fixtures[self.checker.curr_fixture].assign_attrs.add(
AssignPair(node, assign_node)
)
except Exception: # pylint: disable=broad-except
# Cannot find a valid assign expr, skipping the entire attribute
pass
4 changes: 2 additions & 2 deletions pylint_pytest/checkers/fixture.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,14 +8,14 @@
import astroid
import pytest

from ..utils import (
from ..utils.pytest_logic import (
FixtureDict,
_can_use_fixture,
_is_pytest_fixture,
_is_pytest_mark,
_is_pytest_mark_usefixtures,
)
from . import BasePytestChecker
from .types import FixtureDict

# TODO: support pytest python_files configuration
FILE_NAME_PATTERNS: tuple[str, ...] = ("test_*.py", "*_test.py")
Expand Down
7 changes: 0 additions & 7 deletions pylint_pytest/checkers/types.py

This file was deleted.

3 changes: 1 addition & 2 deletions pylint_pytest/checkers/variables.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,8 +6,7 @@
from pylint.checkers.variables import VariablesChecker
from pylint.interfaces import Confidence

from pylint_pytest.utils import _can_use_fixture, _is_same_module

from ..utils.pytest_logic import _can_use_fixture, _is_same_module
from .fixture import FixtureChecker


Expand Down
Loading

0 comments on commit 2066591

Please sign in to comment.