我已经开始使用pathlib.Path
一段时间了,我喜欢使用它。现在我已经习惯了,我变得马虎了,忘记向str
.
tox
将+py.test
与基于tmpdir
(即 a )的临时目录一起使用时,通常会发生这种情况py._path.local.LocalPath
:
from pathlib import Path
import pytest
def test_tmpdir(tmpdir):
p = Path(tmpdir) / 'testfile.csv'
str()
我不是每次都插入,而是更一般地解决这个问题,但不能。
首先,我尝试制作自己的 Path 类,该类具有以下特性_parse_args
:
import pytest
from py._path.local import LocalPath
from pathlib import Path, PurePath
def Path(Path):
@classmethod
def _parse_args(cls, args):
parts = []
for a in args:
if isinstance(a, PurePath):
parts += a._parts
elif isinstance(a, str):
# Force-cast str subclasses to str (issue #21127)
parts.append(str(a))
elif isinstance(a, LocalPath):
parts.append(str(a))
else:
raise TypeError(
"argument should be a path or str object, not %r"
% type(a))
return cls._flavour.parse_parts(parts)
def test_subclass(tmpdir):
p = Path(tmpdir) / 'testfile.csv'
这会抛出一个TypeError: unsupported operand type(s) for /: 'NoneType' and 'str'
(也尝试过PosixPath
,同样的结果,不希望特定于 Linux)。
我试图猴子补丁Path
:
import pytest
from pathlib import Path
def add_tmpdir():
from py._path.local import LocalPath
org_attr = '_parse_args'
stow_attr = '_org_parse_args'
def parse_args_localpath(cls, args):
args = list(args)
for idx, a in enumerate(args):
if isinstance(a, LocalPath):
args[idx] = str(a)
return getattr(cls, stow_attr)(args)
if hasattr(Path, stow_attr):
return # already done
setattr(Path, stow_attr, getattr(Path, org_attr))
setattr(Path, org_attr, parse_args_localpath)
add_tmpdir()
def test_monkeypatch_path(tmpdir):
p = Path(tmpdir) / 'testfile.csv'
这会抛出一个AttributeError: type object 'Path' has no attribute '_flavour'
(也在猴子修补 PurePath 时)。
最后我尝试包装Path
:
import pytest
import pathlib
def Path(*args):
from py._path.local import LocalPath
args = list(args)
for idx, a in enumerate(args):
if isinstance(a, LocalPath):
args[idx] = str(a)
return pathlib.Path(*args)
def test_tmpdir_path(tmpdir):
p = Path(tmpdir) / 'testfile.csv'
这也给出了AttributeError: type object 'Path' has no attribute '_flavour'
我认为在某些时候最后一个有效,但我无法重现。
难道我做错了什么?为什么这么难?