Skip to content

Commit

Permalink
Pytest compat layer
Browse files Browse the repository at this point in the history
  • Loading branch information
iurisilvio committed Feb 27, 2024
1 parent 911b7a0 commit 6d8f700
Show file tree
Hide file tree
Showing 3 changed files with 61 additions and 45 deletions.
45 changes: 10 additions & 35 deletions pytest_ruff.py → pytest_ruff/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,39 +2,13 @@

from pathlib import Path

Check warning on line 3 in pytest_ruff/__init__.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/__init__.py#L3

Added line #L3 was not covered by tests

# Python<=3.8 don't support typing with builtin dict.
from typing import Dict

import pytest

try:
from pytest import StashKey
except ImportError:
from _pytest.store import StoreKey as StashKey

from ruff.__main__ import find_ruff_bin

PYTEST_VER = tuple(int(x) for x in pytest.__version__.split(".")[:2])
from pytest_ruff._pytest_compat import StashKey, get_stash, make_path_kwargs, set_stash

Check warning on line 9 in pytest_ruff/__init__.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/__init__.py#L9

Added line #L9 was not covered by tests

HISTKEY = "ruff/mtimes"
_MTIMES_STASH_KEY = StashKey[Dict[str, float]]()


def _stash(config):
try:
return config.stash
except AttributeError:
return config._store


def _make_path_kwargs(p):
"""
Make keyword arguments passing either path or fspath, depending on pytest version.
In pytest 7.0, the `fspath` argument to Nodes has been deprecated, so we pass `path`
instead.
"""
return dict(path=Path(p)) if PYTEST_VER >= (7, 0) else dict(fspath=p)


def pytest_addoption(parser):
Expand All @@ -51,7 +25,7 @@ def pytest_configure(config):
if not config.option.ruff or not hasattr(config, "cache"):
return

_stash(config)[_MTIMES_STASH_KEY] = config.cache.get(HISTKEY, {})
set_stash(config, config.cache.get(HISTKEY, {}))


def pytest_collect_file(path, parent, fspath=None):

Check warning on line 31 in pytest_ruff/__init__.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/__init__.py#L31

Added line #L31 was not covered by tests
Expand All @@ -62,7 +36,7 @@ def pytest_collect_file(path, parent, fspath=None):
if path.ext != ".py":
return

return RuffFile.from_parent(parent, **_make_path_kwargs(path))
return RuffFile.from_parent(parent, **make_path_kwargs(path))


def pytest_sessionfinish(session, exitstatus):
Expand All @@ -75,7 +49,7 @@ def pytest_sessionfinish(session, exitstatus):
# It works fine if pytest-xdist is not being used.
if not hasattr(config, "workerinput"):
cache = config.cache.get(HISTKEY, {})
cache.update(_stash(config)[_MTIMES_STASH_KEY])
cache.update(get_stash(config))

Check warning on line 52 in pytest_ruff/__init__.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/__init__.py#L52

Added line #L52 was not covered by tests
config.cache.set(HISTKEY, cache)


Expand Down Expand Up @@ -120,16 +94,17 @@ def __init__(self, *k, **kwargs):
self.add_marker("ruff")

def setup(self):
ruffmtimes = _stash(self.config).get(_MTIMES_STASH_KEY, {})
ruffmtimes = get_stash(self.config)
self._ruffmtime = self.fspath.mtime()
old = ruffmtimes.get(str(self.fspath))
if old == self._ruffmtime:
pytest.skip("file previously passed ruff checks")
if ruffmtimes:
old = ruffmtimes.get(str(self.fspath))

Check warning on line 100 in pytest_ruff/__init__.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/__init__.py#L100

Added line #L100 was not covered by tests
if old == self._ruffmtime:
pytest.skip("file previously passed ruff checks")

Check warning on line 102 in pytest_ruff/__init__.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/__init__.py#L102

Added line #L102 was not covered by tests

def runtest(self):
self.handler(path=self.fspath)

ruffmtimes = _stash(self.config).get(_MTIMES_STASH_KEY, None)
ruffmtimes = get_stash(self.config)
if ruffmtimes:
ruffmtimes[str(self.fspath)] = self._ruffmtime

Expand Down
44 changes: 44 additions & 0 deletions pytest_ruff/_pytest_compat.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
from pathlib import Path

Check warning on line 1 in pytest_ruff/_pytest_compat.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/_pytest_compat.py#L1

Added line #L1 was not covered by tests
# Python<=3.8 don't support typing with builtin dict.
from typing import Dict

Check warning on line 3 in pytest_ruff/_pytest_compat.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/_pytest_compat.py#L3

Added line #L3 was not covered by tests

import pytest

Check warning on line 5 in pytest_ruff/_pytest_compat.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/_pytest_compat.py#L5

Added line #L5 was not covered by tests

PYTEST_VER = tuple(int(x) for x in pytest.__version__.split(".")[:2])

try:
from pytest import Stash, StashKey
except ImportError:
from _pytest.store import Store as Stash, StoreKey as StashKey

Check warning on line 12 in pytest_ruff/_pytest_compat.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/_pytest_compat.py#L9-L12

Added lines #L9 - L12 were not covered by tests


_MTIMES_STASH_KEY = StashKey[Dict[str, float]]()

Check warning on line 15 in pytest_ruff/_pytest_compat.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/_pytest_compat.py#L15

Added line #L15 was not covered by tests


def make_path_kwargs(p):

Check warning on line 18 in pytest_ruff/_pytest_compat.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/_pytest_compat.py#L18

Added line #L18 was not covered by tests
"""
Make keyword arguments passing either path or fspath, depending on pytest version.
In pytest 7.0, the `fspath` argument to Nodes has been deprecated, so we pass `path`
instead.
"""
return dict(path=Path(p)) if PYTEST_VER >= (7, 0) else dict(fspath=p)


def get_stash_object(config):

Check warning on line 28 in pytest_ruff/_pytest_compat.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/_pytest_compat.py#L28

Added line #L28 was not covered by tests
try:
stash = config.stash
except AttributeError:
stash = config._store

Check warning on line 32 in pytest_ruff/_pytest_compat.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/_pytest_compat.py#L31-L32

Added lines #L31 - L32 were not covered by tests
return stash


def get_stash(config):

Check warning on line 36 in pytest_ruff/_pytest_compat.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/_pytest_compat.py#L36

Added line #L36 was not covered by tests
missing = object()
stash = get_stash_object(config).get(_MTIMES_STASH_KEY, default=missing)
assert stash is not missing
return stash


def set_stash(config, value):

Check warning on line 43 in pytest_ruff/_pytest_compat.py

View check run for this annotation

Codecov / codecov/patch

pytest_ruff/_pytest_compat.py#L43

Added line #L43 was not covered by tests
get_stash_object(config)[_MTIMES_STASH_KEY] = value
17 changes: 7 additions & 10 deletions tests/test_plugin.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,12 +2,10 @@
import sys

import pytest
from _pytest.config import PytestPluginManager

try:
from pytest import Stash
except ImportError:
from _pytest.store import Store as Stash
import pytest_ruff
from pytest_ruff._pytest_compat import Stash, get_stash


def test_configure(mocker):
Expand All @@ -17,19 +15,18 @@ def test_configure(mocker):
stash=Stash(),
)
pytest_ruff.pytest_configure(config)
assert config.stash[pytest_ruff._MTIMES_STASH_KEY] == mocker.sentinel.cache
assert get_stash(config) == mocker.sentinel.cache


def test_configure_without_ruff(mocker):
config = mocker.Mock(
option=mocker.Mock(ruff=False),
stash=Stash(),
# Mocking to `not hasattr(config, "cache")`.
spec=["addinivalue_line", "option", "stash"],
)
set_stash_mock = mocker.patch("pytest_ruff.set_stash", spec=True)
pytest_ruff.pytest_configure(config)
with pytest.raises(KeyError):
config.stash[pytest_ruff._MTIMES_STASH_KEY]
set_stash_mock.assert_not_called()


def test_check_file():
Expand Down Expand Up @@ -65,7 +62,7 @@ def test_pytest_ruff_format():
stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
).communicate()
assert err == b""
assert err.decode() == ""
assert "File would be reformatted" in out.decode("utf-8")


Expand All @@ -81,5 +78,5 @@ def test_pytest_ruff_noformat():
stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
).communicate()
assert err == b""
assert err.decode() == ""
assert "File would be reformatted" not in out.decode("utf-8")

0 comments on commit 6d8f700

Please sign in to comment.