test_ok2/testing/test_cacheprovider.py

1070 lines
34 KiB
Python
Raw Normal View History

import os
import shutil
import sys
import textwrap
2017-07-07 18:07:33 +08:00
import py
import pytest
from _pytest.main import ExitCode
pytest_plugins = ("pytester",)
2019-06-03 06:32:00 +08:00
class TestNewAPI:
def test_config_cache_makedir(self, testdir):
testdir.makeini("[pytest]")
config = testdir.parseconfigure()
with pytest.raises(ValueError):
config.cache.makedir("key/name")
p = config.cache.makedir("name")
assert p.check()
def test_config_cache_dataerror(self, testdir):
testdir.makeini("[pytest]")
config = testdir.parseconfigure()
cache = config.cache
pytest.raises(TypeError, lambda: cache.set("key/name", cache))
config.cache.set("key/name", 0)
config.cache._getvaluepath("key/name").write_bytes(b"123invalid")
val = config.cache.get("key/name", -2)
assert val == -2
@pytest.mark.filterwarnings("default")
def test_cache_writefail_cachfile_silent(self, testdir):
testdir.makeini("[pytest]")
2018-05-23 22:48:46 +08:00
testdir.tmpdir.join(".pytest_cache").write("gone wrong")
config = testdir.parseconfigure()
cache = config.cache
2018-05-23 22:48:46 +08:00
cache.set("test/broken", [])
2018-05-23 22:48:46 +08:00
@pytest.mark.skipif(sys.platform.startswith("win"), reason="no chmod on windows")
@pytest.mark.filterwarnings(
"ignore:could not create cache path:pytest.PytestWarning"
)
def test_cache_writefail_permissions(self, testdir):
testdir.makeini("[pytest]")
2018-05-23 22:48:46 +08:00
testdir.tmpdir.ensure_dir(".pytest_cache").chmod(0)
config = testdir.parseconfigure()
cache = config.cache
2018-05-23 22:48:46 +08:00
cache.set("test/broken", [])
2018-05-23 22:48:46 +08:00
@pytest.mark.skipif(sys.platform.startswith("win"), reason="no chmod on windows")
2018-09-05 01:44:02 +08:00
@pytest.mark.filterwarnings("default")
def test_cache_failure_warns(self, testdir):
2018-05-23 22:48:46 +08:00
testdir.tmpdir.ensure_dir(".pytest_cache").chmod(0)
testdir.makepyfile(
"""
def test_error():
raise Exception
2018-05-23 22:48:46 +08:00
"""
)
result = testdir.runpytest("-rw")
assert result.ret == 1
2018-10-16 03:39:51 +08:00
# warnings from nodeids, lastfailed, and stepwise
result.stdout.fnmatch_lines(["*could not create cache path*", "*3 warnings*"])
def test_config_cache(self, testdir):
2018-05-23 22:48:46 +08:00
testdir.makeconftest(
"""
def pytest_configure(config):
# see that we get cache information early on
assert hasattr(config, "cache")
2018-05-23 22:48:46 +08:00
"""
)
testdir.makepyfile(
"""
def test_session(pytestconfig):
assert hasattr(pytestconfig, "cache")
2018-05-23 22:48:46 +08:00
"""
)
result = testdir.runpytest()
assert result.ret == 0
result.stdout.fnmatch_lines(["*1 passed*"])
def test_cachefuncarg(self, testdir):
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
"""
import pytest
def test_cachefuncarg(cache):
val = cache.get("some/thing", None)
assert val is None
cache.set("some/thing", [1])
pytest.raises(TypeError, lambda: cache.get("some/thing"))
val = cache.get("some/thing", [])
assert val == [1]
2018-05-23 22:48:46 +08:00
"""
)
result = testdir.runpytest()
assert result.ret == 0
result.stdout.fnmatch_lines(["*1 passed*"])
2017-07-07 18:07:33 +08:00
def test_custom_rel_cache_dir(self, testdir):
2018-05-23 22:48:46 +08:00
rel_cache_dir = os.path.join("custom_cache_dir", "subdir")
testdir.makeini(
"""
2017-07-07 18:07:33 +08:00
[pytest]
cache_dir = {cache_dir}
2018-05-23 22:48:46 +08:00
""".format(
cache_dir=rel_cache_dir
)
)
testdir.makepyfile(test_errored="def test_error():\n assert False")
2017-07-07 18:07:33 +08:00
testdir.runpytest()
assert testdir.tmpdir.join(rel_cache_dir).isdir()
def test_custom_abs_cache_dir(self, testdir, tmpdir_factory):
2018-05-23 22:48:46 +08:00
tmp = str(tmpdir_factory.mktemp("tmp"))
abs_cache_dir = os.path.join(tmp, "custom_cache_dir")
testdir.makeini(
"""
2017-07-07 18:07:33 +08:00
[pytest]
cache_dir = {cache_dir}
2018-05-23 22:48:46 +08:00
""".format(
cache_dir=abs_cache_dir
)
)
testdir.makepyfile(test_errored="def test_error():\n assert False")
2017-07-07 18:07:33 +08:00
testdir.runpytest()
assert py.path.local(abs_cache_dir).isdir()
def test_custom_cache_dir_with_env_var(self, testdir, monkeypatch):
2018-05-23 22:48:46 +08:00
monkeypatch.setenv("env_var", "custom_cache_dir")
testdir.makeini(
"""
2017-07-07 18:07:33 +08:00
[pytest]
cache_dir = {cache_dir}
2018-05-23 22:48:46 +08:00
""".format(
cache_dir="$env_var"
)
)
testdir.makepyfile(test_errored="def test_error():\n assert False")
2017-07-07 18:07:33 +08:00
testdir.runpytest()
2018-05-23 22:48:46 +08:00
assert testdir.tmpdir.join("custom_cache_dir").isdir()
2017-07-18 08:16:14 +08:00
@pytest.mark.parametrize("env", ((), ("TOX_ENV_DIR", "/tox_env_dir")))
def test_cache_reportheader(env, testdir, monkeypatch):
testdir.makepyfile("""def test_foo(): pass""")
if env:
monkeypatch.setenv(*env)
expected = os.path.join(env[1], ".pytest_cache")
else:
monkeypatch.delenv("TOX_ENV_DIR", raising=False)
expected = ".pytest_cache"
result = testdir.runpytest("-v")
result.stdout.fnmatch_lines(["cachedir: %s" % expected])
def test_cache_reportheader_external_abspath(testdir, tmpdir_factory):
external_cache = tmpdir_factory.mktemp(
"test_cache_reportheader_external_abspath_abs"
)
testdir.makepyfile(
"""
def test_hello():
pass
"""
)
testdir.makeini(
"""
[pytest]
cache_dir = {abscache}
""".format(
abscache=external_cache
)
)
result = testdir.runpytest("-v")
result.stdout.fnmatch_lines(
["cachedir: {abscache}".format(abscache=external_cache)]
)
def test_cache_show(testdir):
result = testdir.runpytest("--cache-show")
assert result.ret == 0
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["*cache is empty*"])
testdir.makeconftest(
"""
def pytest_configure(config):
config.cache.set("my/name", [1,2,3])
config.cache.set("my/hello", "world")
config.cache.set("other/some", {1:2})
dp = config.cache.makedir("mydb")
dp.ensure("hello")
dp.ensure("world")
2018-05-23 22:48:46 +08:00
"""
)
result = testdir.runpytest()
assert result.ret == 5 # no tests executed
result = testdir.runpytest("--cache-show")
result.stdout.fnmatch_lines(
2018-05-23 22:48:46 +08:00
[
"*cachedir:*",
"*- cache values for '[*]' -*",
"cache/nodeids contains:",
"my/name contains:",
2018-05-23 22:48:46 +08:00
" [1, 2, 3]",
"other/some contains:",
" {*'1': 2}",
"*- cache directories for '[*]' -*",
2018-05-23 22:48:46 +08:00
"*mydb/hello*length 0*",
"*mydb/world*length 0*",
]
)
assert result.ret == 0
result = testdir.runpytest("--cache-show", "*/hello")
result.stdout.fnmatch_lines(
[
"*cachedir:*",
"*- cache values for '[*]/hello' -*",
"my/hello contains:",
" *'world'",
"*- cache directories for '[*]/hello' -*",
"d/mydb/hello*length 0*",
]
)
stdout = result.stdout.str()
assert "other/some" not in stdout
assert "d/mydb/world" not in stdout
assert result.ret == 0
2019-06-03 06:32:00 +08:00
class TestLastFailed:
def test_lastfailed_usecase(self, testdir, monkeypatch):
monkeypatch.setenv("PYTHONDONTWRITEBYTECODE", "1")
2018-05-23 22:48:46 +08:00
p = testdir.makepyfile(
"""
def test_1():
assert 0
def test_2():
assert 0
def test_3():
assert 1
2018-05-23 22:48:46 +08:00
"""
)
result = testdir.runpytest()
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["*2 failed*"])
p.write(
textwrap.dedent(
"""\
def test_1():
assert 1
def test_2():
assert 1
def test_3():
assert 0
"""
2018-05-23 22:48:46 +08:00
)
)
result = testdir.runpytest("--lf")
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["*2 passed*1 desel*"])
result = testdir.runpytest("--lf")
2019-04-03 21:18:29 +08:00
result.stdout.fnmatch_lines(
[
"collected 3 items",
"run-last-failure: no previously failed tests, not deselecting items.",
"*1 failed*2 passed*",
]
)
result = testdir.runpytest("--lf", "--cache-clear")
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["*1 failed*2 passed*"])
# Run this again to make sure clear-cache is robust
2018-05-23 22:48:46 +08:00
if os.path.isdir(".pytest_cache"):
shutil.rmtree(".pytest_cache")
result = testdir.runpytest("--lf", "--cache-clear")
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["*1 failed*2 passed*"])
def test_failedfirst_order(self, testdir):
2018-05-23 22:48:46 +08:00
testdir.tmpdir.join("test_a.py").write(
textwrap.dedent(
"""\
def test_always_passes():
assert 1
2018-05-23 22:48:46 +08:00
"""
)
)
testdir.tmpdir.join("test_b.py").write(
textwrap.dedent(
"""\
def test_always_fails():
assert 0
2018-05-23 22:48:46 +08:00
"""
)
)
result = testdir.runpytest()
# Test order will be collection order; alphabetical
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["test_a.py*", "test_b.py*"])
result = testdir.runpytest("--ff")
# Test order will be failing tests firs
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["test_b.py*", "test_a.py*"])
def test_lastfailed_failedfirst_order(self, testdir):
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
**{
"test_a.py": """\
def test_always_passes():
assert 1
""",
"test_b.py": """\
def test_always_fails():
assert 0
""",
2018-05-23 22:48:46 +08:00
}
)
result = testdir.runpytest()
# Test order will be collection order; alphabetical
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["test_a.py*", "test_b.py*"])
result = testdir.runpytest("--lf", "--ff")
# Test order will be failing tests firs
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["test_b.py*"])
assert "test_a.py" not in result.stdout.str()
def test_lastfailed_difference_invocations(self, testdir, monkeypatch):
monkeypatch.setenv("PYTHONDONTWRITEBYTECODE", "1")
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
test_a="""\
def test_a1():
assert 0
def test_a2():
assert 1
""",
test_b="""\
def test_b1():
assert 0
""",
2018-05-23 22:48:46 +08:00
)
p = testdir.tmpdir.join("test_a.py")
p2 = testdir.tmpdir.join("test_b.py")
result = testdir.runpytest()
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["*2 failed*"])
result = testdir.runpytest("--lf", p2)
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["*1 failed*"])
p2.write(
textwrap.dedent(
"""\
def test_b1():
assert 1
2018-05-23 22:48:46 +08:00
"""
)
)
result = testdir.runpytest("--lf", p2)
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["*1 passed*"])
result = testdir.runpytest("--lf", p)
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["*1 failed*1 desel*"])
def test_lastfailed_usecase_splice(self, testdir, monkeypatch):
monkeypatch.setenv("PYTHONDONTWRITEBYTECODE", "1")
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
"""\
def test_1():
assert 0
"""
2018-05-23 22:48:46 +08:00
)
p2 = testdir.tmpdir.join("test_something.py")
2018-05-23 22:48:46 +08:00
p2.write(
textwrap.dedent(
"""\
def test_2():
assert 0
2018-05-23 22:48:46 +08:00
"""
)
)
result = testdir.runpytest()
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["*2 failed*"])
result = testdir.runpytest("--lf", p2)
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["*1 failed*"])
result = testdir.runpytest("--lf")
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["*2 failed*"])
def test_lastfailed_xpass(self, testdir):
2018-05-23 22:48:46 +08:00
testdir.inline_runsource(
"""
import pytest
@pytest.mark.xfail
def test_hello():
assert 1
2018-05-23 22:48:46 +08:00
"""
)
config = testdir.parseconfigure()
lastfailed = config.cache.get("cache/lastfailed", -1)
assert lastfailed == -1
def test_non_serializable_parametrize(self, testdir):
"""Test that failed parametrized tests with unmarshable parameters
don't break pytest-cache.
"""
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
r"""
import pytest
@pytest.mark.parametrize('val', [
b'\xac\x10\x02G',
])
def test_fail(val):
assert False
2018-05-23 22:48:46 +08:00
"""
)
result = testdir.runpytest()
result.stdout.fnmatch_lines(["*1 failed in*"])
def test_terminal_report_lastfailed(self, testdir):
2018-05-23 22:48:46 +08:00
test_a = testdir.makepyfile(
test_a="""
def test_a1():
pass
def test_a2():
pass
2018-05-23 22:48:46 +08:00
"""
)
test_b = testdir.makepyfile(
test_b="""
def test_b1():
assert 0
def test_b2():
assert 0
2018-05-23 22:48:46 +08:00
"""
)
result = testdir.runpytest()
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["collected 4 items", "*2 failed, 2 passed in*"])
result = testdir.runpytest("--lf")
result.stdout.fnmatch_lines(
[
"collected 2 items",
"run-last-failure: rerun previous 2 failures (skipped 1 file)",
"*2 failed in*",
2018-05-23 22:48:46 +08:00
]
)
result = testdir.runpytest(test_a, "--lf")
2019-04-03 21:18:29 +08:00
result.stdout.fnmatch_lines(
[
"collected 2 items",
"run-last-failure: 2 known failures not in selected tests",
"*2 passed in*",
]
)
2018-05-23 22:48:46 +08:00
result = testdir.runpytest(test_b, "--lf")
result.stdout.fnmatch_lines(
[
"collected 2 items",
"run-last-failure: rerun previous 2 failures",
"*2 failed in*",
]
)
result = testdir.runpytest("test_b.py::test_b1", "--lf")
result.stdout.fnmatch_lines(
[
"collected 1 item",
"run-last-failure: rerun previous 1 failure",
"*1 failed in*",
]
)
def test_terminal_report_failedfirst(self, testdir):
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
test_a="""
def test_a1():
assert 0
def test_a2():
pass
2018-05-23 22:48:46 +08:00
"""
)
result = testdir.runpytest()
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["collected 2 items", "*1 failed, 1 passed in*"])
result = testdir.runpytest("--ff")
result.stdout.fnmatch_lines(
[
"collected 2 items",
"run-last-failure: rerun previous 1 failure first",
"*1 failed, 1 passed in*",
]
)
def test_lastfailed_collectfailure(self, testdir, monkeypatch):
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
test_maybe="""
2017-12-27 11:47:26 +08:00
import os
env = os.environ
if '1' == env['FAILIMPORT']:
raise ImportError('fail')
def test_hello():
assert '0' == env['FAILTEST']
2018-05-23 22:48:46 +08:00
"""
)
def rlf(fail_import, fail_run):
monkeypatch.setenv("FAILIMPORT", str(fail_import))
monkeypatch.setenv("FAILTEST", str(fail_run))
2018-05-23 22:48:46 +08:00
testdir.runpytest("-q")
config = testdir.parseconfigure()
lastfailed = config.cache.get("cache/lastfailed", -1)
return lastfailed
lastfailed = rlf(fail_import=0, fail_run=0)
assert lastfailed == -1
lastfailed = rlf(fail_import=1, fail_run=0)
2018-05-23 22:48:46 +08:00
assert list(lastfailed) == ["test_maybe.py"]
lastfailed = rlf(fail_import=0, fail_run=1)
2018-05-23 22:48:46 +08:00
assert list(lastfailed) == ["test_maybe.py::test_hello"]
def test_lastfailed_failure_subset(self, testdir, monkeypatch):
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
test_maybe="""
2017-12-27 11:47:26 +08:00
import os
env = os.environ
if '1' == env['FAILIMPORT']:
raise ImportError('fail')
def test_hello():
assert '0' == env['FAILTEST']
2018-05-23 22:48:46 +08:00
"""
)
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
test_maybe2="""
2017-12-27 11:47:26 +08:00
import os
env = os.environ
if '1' == env['FAILIMPORT']:
raise ImportError('fail')
def test_hello():
assert '0' == env['FAILTEST']
def test_pass():
pass
2018-05-23 22:48:46 +08:00
"""
)
def rlf(fail_import, fail_run, args=()):
monkeypatch.setenv("FAILIMPORT", str(fail_import))
monkeypatch.setenv("FAILTEST", str(fail_run))
2018-05-23 22:48:46 +08:00
result = testdir.runpytest("-q", "--lf", *args)
config = testdir.parseconfigure()
lastfailed = config.cache.get("cache/lastfailed", -1)
return result, lastfailed
result, lastfailed = rlf(fail_import=0, fail_run=0)
assert lastfailed == -1
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(["*3 passed*"])
result, lastfailed = rlf(fail_import=1, fail_run=0)
2018-05-23 22:48:46 +08:00
assert sorted(list(lastfailed)) == ["test_maybe.py", "test_maybe2.py"]
2018-05-23 22:48:46 +08:00
result, lastfailed = rlf(fail_import=0, fail_run=0, args=("test_maybe2.py",))
assert list(lastfailed) == ["test_maybe.py"]
# edge case of test selection - even if we remember failures
# from other tests we still need to run all tests if no test
# matches the failures
2018-05-23 22:48:46 +08:00
result, lastfailed = rlf(fail_import=0, fail_run=0, args=("test_maybe2.py",))
assert list(lastfailed) == ["test_maybe.py"]
result.stdout.fnmatch_lines(["*2 passed*"])
def test_lastfailed_creates_cache_when_needed(self, testdir):
# Issue #1342
2018-05-23 22:48:46 +08:00
testdir.makepyfile(test_empty="")
testdir.runpytest("-q", "--lf")
assert not os.path.exists(".pytest_cache/v/cache/lastfailed")
2018-05-23 22:48:46 +08:00
testdir.makepyfile(test_successful="def test_success():\n assert True")
testdir.runpytest("-q", "--lf")
assert not os.path.exists(".pytest_cache/v/cache/lastfailed")
2018-05-23 22:48:46 +08:00
testdir.makepyfile(test_errored="def test_error():\n assert False")
testdir.runpytest("-q", "--lf")
assert os.path.exists(".pytest_cache/v/cache/lastfailed")
2017-07-27 21:56:04 +08:00
def test_xfail_not_considered_failure(self, testdir):
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
"""
2017-07-27 21:56:04 +08:00
import pytest
@pytest.mark.xfail
def test():
assert 0
2018-05-23 22:48:46 +08:00
"""
)
2017-07-27 21:56:04 +08:00
result = testdir.runpytest()
result.stdout.fnmatch_lines(["*1 xfailed*"])
2017-07-27 21:56:04 +08:00
assert self.get_cached_last_failed(testdir) == []
def test_xfail_strict_considered_failure(self, testdir):
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
"""
2017-07-27 21:56:04 +08:00
import pytest
@pytest.mark.xfail(strict=True)
def test():
pass
2018-05-23 22:48:46 +08:00
"""
)
2017-07-27 21:56:04 +08:00
result = testdir.runpytest()
result.stdout.fnmatch_lines(["*1 failed*"])
assert self.get_cached_last_failed(testdir) == [
"test_xfail_strict_considered_failure.py::test"
]
2017-07-27 21:56:04 +08:00
2018-05-23 22:48:46 +08:00
@pytest.mark.parametrize("mark", ["mark.xfail", "mark.skip"])
def test_failed_changed_to_xfail_or_skip(self, testdir, mark):
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
"""
import pytest
def test():
assert 0
2018-05-23 22:48:46 +08:00
"""
)
result = testdir.runpytest()
assert self.get_cached_last_failed(testdir) == [
"test_failed_changed_to_xfail_or_skip.py::test"
]
assert result.ret == 1
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
"""
import pytest
@pytest.{mark}
def test():
assert 0
2018-05-23 22:48:46 +08:00
""".format(
mark=mark
)
)
result = testdir.runpytest()
assert result.ret == 0
assert self.get_cached_last_failed(testdir) == []
assert result.ret == 0
@pytest.mark.parametrize("quiet", [True, False])
@pytest.mark.parametrize("opt", ["--ff", "--lf"])
def test_lf_and_ff_prints_no_needless_message(self, quiet, opt, testdir):
# Issue 3853
testdir.makepyfile("def test(): assert 0")
args = [opt]
if quiet:
args.append("-q")
result = testdir.runpytest(*args)
assert "run all" not in result.stdout.str()
result = testdir.runpytest(*args)
if quiet:
assert "run all" not in result.stdout.str()
else:
assert "rerun previous" in result.stdout.str()
def get_cached_last_failed(self, testdir):
config = testdir.parseconfigure()
return sorted(config.cache.get("cache/lastfailed", {}))
def test_cache_cumulative(self, testdir):
"""
Test workflow where user fixes errors gradually file by file using --lf.
"""
# 1. initial run
2018-05-23 22:48:46 +08:00
test_bar = testdir.makepyfile(
test_bar="""
def test_bar_1():
pass
def test_bar_2():
assert 0
2018-05-23 22:48:46 +08:00
"""
)
test_foo = testdir.makepyfile(
test_foo="""
def test_foo_3():
pass
def test_foo_4():
assert 0
2018-05-23 22:48:46 +08:00
"""
)
testdir.runpytest()
assert self.get_cached_last_failed(testdir) == [
"test_bar.py::test_bar_2",
"test_foo.py::test_foo_4",
]
# 2. fix test_bar_2, run only test_bar.py
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
test_bar="""
def test_bar_1():
pass
def test_bar_2():
pass
2018-05-23 22:48:46 +08:00
"""
)
result = testdir.runpytest(test_bar)
result.stdout.fnmatch_lines(["*2 passed*"])
# ensure cache does not forget that test_foo_4 failed once before
2018-05-23 22:48:46 +08:00
assert self.get_cached_last_failed(testdir) == ["test_foo.py::test_foo_4"]
2018-05-23 22:48:46 +08:00
result = testdir.runpytest("--last-failed")
result.stdout.fnmatch_lines(["*1 failed, 1 deselected*"])
2018-05-23 22:48:46 +08:00
assert self.get_cached_last_failed(testdir) == ["test_foo.py::test_foo_4"]
# 3. fix test_foo_4, run only test_foo.py
2018-05-23 22:48:46 +08:00
test_foo = testdir.makepyfile(
test_foo="""
def test_foo_3():
pass
def test_foo_4():
pass
2018-05-23 22:48:46 +08:00
"""
)
result = testdir.runpytest(test_foo, "--last-failed")
result.stdout.fnmatch_lines(["*1 passed, 1 deselected*"])
assert self.get_cached_last_failed(testdir) == []
2018-05-23 22:48:46 +08:00
result = testdir.runpytest("--last-failed")
result.stdout.fnmatch_lines(["*4 passed*"])
assert self.get_cached_last_failed(testdir) == []
2018-02-14 03:49:28 +08:00
2018-03-11 04:45:45 +08:00
def test_lastfailed_no_failures_behavior_all_passed(self, testdir):
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
"""
2018-03-11 04:45:45 +08:00
def test_1():
assert True
def test_2():
assert True
2018-05-23 22:48:46 +08:00
"""
)
2018-03-11 04:45:45 +08:00
result = testdir.runpytest()
result.stdout.fnmatch_lines(["*2 passed*"])
result = testdir.runpytest("--lf")
result.stdout.fnmatch_lines(["*2 passed*"])
result = testdir.runpytest("--lf", "--lfnf", "all")
result.stdout.fnmatch_lines(["*2 passed*"])
result = testdir.runpytest("--lf", "--lfnf", "none")
2019-04-03 21:18:29 +08:00
result.stdout.fnmatch_lines(
[
"collected 2 items / 2 deselected",
"run-last-failure: no previously failed tests, deselecting all items.",
"* 2 deselected in *",
]
)
assert result.ret == ExitCode.NO_TESTS_COLLECTED
2018-03-11 04:45:45 +08:00
def test_lastfailed_no_failures_behavior_empty_cache(self, testdir):
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
"""
2018-03-11 04:45:45 +08:00
def test_1():
assert True
def test_2():
assert False
2018-05-23 22:48:46 +08:00
"""
)
2018-03-11 04:45:45 +08:00
result = testdir.runpytest("--lf", "--cache-clear")
result.stdout.fnmatch_lines(["*1 failed*1 passed*"])
result = testdir.runpytest("--lf", "--cache-clear", "--lfnf", "all")
result.stdout.fnmatch_lines(["*1 failed*1 passed*"])
result = testdir.runpytest("--lf", "--cache-clear", "--lfnf", "none")
result.stdout.fnmatch_lines(["*2 desel*"])
def test_lastfailed_skip_collection(self, testdir):
"""
Test --lf behavior regarding skipping collection of files that are not marked as
failed in the cache (#5172).
"""
testdir.makepyfile(
**{
"pkg1/test_1.py": """
import pytest
@pytest.mark.parametrize('i', range(3))
def test_1(i): pass
""",
"pkg2/test_2.py": """
import pytest
@pytest.mark.parametrize('i', range(5))
def test_1(i):
assert i not in (1, 3)
""",
}
)
# first run: collects 8 items (test_1: 3, test_2: 5)
result = testdir.runpytest()
result.stdout.fnmatch_lines(["collected 8 items", "*2 failed*6 passed*"])
# second run: collects only 5 items from test_2, because all tests from test_1 have passed
result = testdir.runpytest("--lf")
result.stdout.fnmatch_lines(
[
"collected 5 items / 3 deselected / 2 selected",
"run-last-failure: rerun previous 2 failures (skipped 1 file)",
"*2 failed*3 deselected*",
]
)
# add another file and check if message is correct when skipping more than 1 file
testdir.makepyfile(
**{
"pkg1/test_3.py": """
def test_3(): pass
"""
}
)
result = testdir.runpytest("--lf")
result.stdout.fnmatch_lines(
[
"collected 5 items / 3 deselected / 2 selected",
"run-last-failure: rerun previous 2 failures (skipped 2 files)",
"*2 failed*3 deselected*",
]
)
def test_lastfailed_with_known_failures_not_being_selected(self, testdir):
testdir.makepyfile(
**{
"pkg1/test_1.py": """def test_1(): assert 0""",
"pkg1/test_2.py": """def test_2(): pass""",
}
)
result = testdir.runpytest()
result.stdout.fnmatch_lines(["collected 2 items", "* 1 failed, 1 passed in *"])
py.path.local("pkg1/test_1.py").remove()
result = testdir.runpytest("--lf")
result.stdout.fnmatch_lines(
[
"collected 1 item",
"run-last-failure: 1 known failures not in selected tests",
"* 1 passed in *",
]
)
# Recreate file with known failure.
testdir.makepyfile(**{"pkg1/test_1.py": """def test_1(): assert 0"""})
result = testdir.runpytest("--lf")
result.stdout.fnmatch_lines(
[
"collected 1 item",
"run-last-failure: rerun previous 1 failure (skipped 1 file)",
"* 1 failed in *",
]
)
# Remove/rename test.
testdir.makepyfile(**{"pkg1/test_1.py": """def test_renamed(): assert 0"""})
result = testdir.runpytest("--lf")
result.stdout.fnmatch_lines(
[
"collected 1 item",
"run-last-failure: 1 known failures not in selected tests (skipped 1 file)",
"* 1 failed in *",
]
)
2018-02-14 03:49:28 +08:00
2019-06-03 06:32:00 +08:00
class TestNewFirst:
2018-02-14 03:49:28 +08:00
def test_newfirst_usecase(self, testdir):
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
**{
"test_1/test_1.py": """
2018-02-24 03:49:17 +08:00
def test_1(): assert 1
def test_2(): assert 1
def test_3(): assert 1
2018-05-23 22:48:46 +08:00
""",
"test_2/test_2.py": """
2018-02-24 03:49:17 +08:00
def test_1(): assert 1
def test_2(): assert 1
def test_3(): assert 1
2018-05-23 22:48:46 +08:00
""",
}
)
2018-02-14 03:49:28 +08:00
2018-05-23 22:48:46 +08:00
testdir.tmpdir.join("test_1/test_1.py").setmtime(1)
2018-02-14 03:49:28 +08:00
result = testdir.runpytest("-v")
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(
[
"*test_1/test_1.py::test_1 PASSED*",
"*test_1/test_1.py::test_2 PASSED*",
"*test_1/test_1.py::test_3 PASSED*",
"*test_2/test_2.py::test_1 PASSED*",
"*test_2/test_2.py::test_2 PASSED*",
"*test_2/test_2.py::test_3 PASSED*",
]
)
2018-02-14 03:49:28 +08:00
result = testdir.runpytest("-v", "--nf")
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(
[
"*test_2/test_2.py::test_1 PASSED*",
"*test_2/test_2.py::test_2 PASSED*",
"*test_2/test_2.py::test_3 PASSED*",
"*test_1/test_1.py::test_1 PASSED*",
"*test_1/test_1.py::test_2 PASSED*",
"*test_1/test_1.py::test_3 PASSED*",
]
)
2018-02-14 03:49:28 +08:00
2018-02-24 03:49:17 +08:00
testdir.tmpdir.join("test_1/test_1.py").write(
2018-02-14 03:49:28 +08:00
"def test_1(): assert 1\n"
"def test_2(): assert 1\n"
"def test_3(): assert 1\n"
"def test_4(): assert 1\n"
)
2018-05-23 22:48:46 +08:00
testdir.tmpdir.join("test_1/test_1.py").setmtime(1)
2018-02-14 03:49:28 +08:00
result = testdir.runpytest("-v", "--nf")
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(
[
"*test_1/test_1.py::test_4 PASSED*",
"*test_2/test_2.py::test_1 PASSED*",
"*test_2/test_2.py::test_2 PASSED*",
"*test_2/test_2.py::test_3 PASSED*",
"*test_1/test_1.py::test_1 PASSED*",
"*test_1/test_1.py::test_2 PASSED*",
"*test_1/test_1.py::test_3 PASSED*",
]
)
2018-02-14 03:49:28 +08:00
def test_newfirst_parametrize(self, testdir):
2018-05-23 22:48:46 +08:00
testdir.makepyfile(
**{
"test_1/test_1.py": """
2018-02-24 03:49:17 +08:00
import pytest
@pytest.mark.parametrize('num', [1, 2])
def test_1(num): assert num
2018-05-23 22:48:46 +08:00
""",
"test_2/test_2.py": """
2018-02-24 03:49:17 +08:00
import pytest
@pytest.mark.parametrize('num', [1, 2])
def test_1(num): assert num
2018-05-23 22:48:46 +08:00
""",
}
)
2018-02-14 03:49:28 +08:00
2018-05-23 22:48:46 +08:00
testdir.tmpdir.join("test_1/test_1.py").setmtime(1)
2018-02-14 03:49:28 +08:00
result = testdir.runpytest("-v")
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(
[
"*test_1/test_1.py::test_1[1*",
"*test_1/test_1.py::test_1[2*",
"*test_2/test_2.py::test_1[1*",
"*test_2/test_2.py::test_1[2*",
]
)
2018-02-14 03:49:28 +08:00
result = testdir.runpytest("-v", "--nf")
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(
[
"*test_2/test_2.py::test_1[1*",
"*test_2/test_2.py::test_1[2*",
"*test_1/test_1.py::test_1[1*",
"*test_1/test_1.py::test_1[2*",
]
)
2018-02-14 03:49:28 +08:00
2018-02-24 03:49:17 +08:00
testdir.tmpdir.join("test_1/test_1.py").write(
2018-02-14 03:49:28 +08:00
"import pytest\n"
"@pytest.mark.parametrize('num', [1, 2, 3])\n"
"def test_1(num): assert num\n"
)
2018-05-23 22:48:46 +08:00
testdir.tmpdir.join("test_1/test_1.py").setmtime(1)
2018-02-14 03:49:28 +08:00
result = testdir.runpytest("-v", "--nf")
2018-05-23 22:48:46 +08:00
result.stdout.fnmatch_lines(
[
"*test_1/test_1.py::test_1[3*",
"*test_2/test_2.py::test_1[1*",
"*test_2/test_2.py::test_1[2*",
"*test_1/test_1.py::test_1[1*",
"*test_1/test_1.py::test_1[2*",
]
)
2019-06-03 06:32:00 +08:00
class TestReadme:
def check_readme(self, testdir):
config = testdir.parseconfigure()
2018-06-23 06:03:10 +08:00
readme = config.cache._cachedir.joinpath("README.md")
return readme.is_file()
def test_readme_passed(self, testdir):
testdir.makepyfile(
"""
def test_always_passes():
assert 1
"""
)
testdir.runpytest()
assert self.check_readme(testdir) is True
def test_readme_failed(self, testdir):
testdir.makepyfile(
"""
2019-03-04 23:42:51 +08:00
def test_always_fails():
assert 0
"""
)
testdir.runpytest()
assert self.check_readme(testdir) is True
def test_gitignore(testdir):
"""Ensure we automatically create .gitignore file in the pytest_cache directory (#3286)."""
from _pytest.cacheprovider import Cache
config = testdir.parseconfig()
cache = Cache.for_config(config)
cache.set("foo", "bar")
msg = "# Created by pytest automatically.\n*"
gitignore_path = cache._cachedir.joinpath(".gitignore")
assert gitignore_path.read_text(encoding="UTF-8") == msg
# Does not overwrite existing/custom one.
2019-06-03 06:32:00 +08:00
gitignore_path.write_text("custom")
cache.set("something", "else")
2018-11-23 07:38:33 +08:00
assert gitignore_path.read_text(encoding="UTF-8") == "custom"
def test_does_not_create_boilerplate_in_existing_dirs(testdir):
from _pytest.cacheprovider import Cache
testdir.makeini(
"""
[pytest]
cache_dir = .
"""
)
config = testdir.parseconfig()
cache = Cache.for_config(config)
cache.set("foo", "bar")
2018-11-23 07:38:33 +08:00
assert os.path.isdir("v") # cache contents
assert not os.path.exists(".gitignore")
assert not os.path.exists("README.md")
def test_cachedir_tag(testdir):
"""Ensure we automatically create CACHEDIR.TAG file in the pytest_cache directory (#4278)."""
from _pytest.cacheprovider import Cache
from _pytest.cacheprovider import CACHEDIR_TAG_CONTENT
config = testdir.parseconfig()
cache = Cache.for_config(config)
cache.set("foo", "bar")
cachedir_tag_path = cache._cachedir.joinpath("CACHEDIR.TAG")
assert cachedir_tag_path.read_bytes() == CACHEDIR_TAG_CONTENT