tmpdir_factory.mktemp now fails given absolute and non-normaliz… (#6323)

tmpdir_factory.mktemp now fails given absolute and non-normalized paths.
This commit is contained in:
Bruno Oliveira 2020-01-09 19:10:03 -03:00 committed by GitHub
commit 29db2da9a7
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 53 additions and 12 deletions

View File

@ -0,0 +1 @@
``tmpdir_factory.mktemp`` now fails when given absolute and non-normalized paths.

View File

@ -45,8 +45,30 @@ class TempPathFactory:
given_basetemp=config.option.basetemp, trace=config.trace.get("tmpdir") given_basetemp=config.option.basetemp, trace=config.trace.get("tmpdir")
) )
def _ensure_relative_to_basetemp(self, basename: str):
basename = os.path.normpath(basename)
if (self.getbasetemp() / basename).resolve().parent != self.getbasetemp():
raise ValueError(
"{} is not a normalized and relative path".format(basename)
)
return basename
def mktemp(self, basename: str, numbered: bool = True) -> Path: def mktemp(self, basename: str, numbered: bool = True) -> Path:
"""makes a temporary directory managed by the factory""" """Creates a new temporary directory managed by the factory.
:param basename:
Directory base name, must be a relative path.
:param numbered:
If True, ensure the directory is unique by adding a number
prefix greater than any existing one: ``basename="foo"`` and ``numbered=True``
means that this function will create directories named ``"foo-0"``,
``"foo-1"``, ``"foo-2"`` and so on.
:return:
The path to the new directory.
"""
basename = self._ensure_relative_to_basetemp(basename)
if not numbered: if not numbered:
p = self.getbasetemp().joinpath(basename) p = self.getbasetemp().joinpath(basename)
p.mkdir() p.mkdir()
@ -90,10 +112,9 @@ class TempdirFactory:
_tmppath_factory = attr.ib(type=TempPathFactory) _tmppath_factory = attr.ib(type=TempPathFactory)
def mktemp(self, basename: str, numbered: bool = True): def mktemp(self, basename: str, numbered: bool = True) -> py.path.local:
"""Create a subdirectory of the base temporary directory and return it. """
If ``numbered``, ensure the directory is unique by adding a number Same as :meth:`TempPathFactory.mkdir`, but returns a ``py.path.local`` object.
prefix greater than any existing one.
""" """
return py.path.local(self._tmppath_factory.mktemp(basename, numbered).resolve()) return py.path.local(self._tmppath_factory.mktemp(basename, numbered).resolve())

View File

@ -74,19 +74,38 @@ class TestConfigTmpdir:
assert not mytemp.join("hello").check() assert not mytemp.join("hello").check()
def test_basetemp(testdir): testdata = [
("mypath", True),
("/mypath1", False),
("./mypath1", True),
("../mypath3", False),
("../../mypath4", False),
("mypath5/..", False),
("mypath6/../mypath6", True),
("mypath7/../mypath7/..", False),
]
@pytest.mark.parametrize("basename, is_ok", testdata)
def test_mktemp(testdir, basename, is_ok):
mytemp = testdir.tmpdir.mkdir("mytemp") mytemp = testdir.tmpdir.mkdir("mytemp")
p = testdir.makepyfile( p = testdir.makepyfile(
""" """
import pytest import pytest
def test_1(tmpdir_factory): def test_abs_path(tmpdir_factory):
tmpdir_factory.mktemp('hello', numbered=False) tmpdir_factory.mktemp('{}', numbered=False)
""" """.format(
basename
) )
)
result = testdir.runpytest(p, "--basetemp=%s" % mytemp) result = testdir.runpytest(p, "--basetemp=%s" % mytemp)
if is_ok:
assert result.ret == 0 assert result.ret == 0
print(mytemp) assert mytemp.join(basename).check()
assert mytemp.join("hello").check() else:
assert result.ret == 1
result.stdout.fnmatch_lines("*ValueError*")
def test_tmpdir_always_is_realpath(testdir): def test_tmpdir_always_is_realpath(testdir):