111 lines
3.7 KiB
Python
111 lines
3.7 KiB
Python
import pytest
|
|
import sys
|
|
|
|
pytest_plugins = "pytester",
|
|
|
|
import os, py
|
|
pid = os.getpid()
|
|
|
|
def pytest_addoption(parser):
|
|
parser.addoption('--lsof',
|
|
action="store_true", dest="lsof", default=False,
|
|
help=("run FD checks if lsof is available"))
|
|
|
|
def pytest_configure(config):
|
|
config.addinivalue_line("markers",
|
|
"multi(arg=[value1,value2, ...]): call the test function "
|
|
"multiple times with arg=value1, then with arg=value2, ... "
|
|
)
|
|
if config.getvalue("lsof"):
|
|
try:
|
|
out = py.process.cmdexec("lsof -p %d" % pid)
|
|
except py.process.cmdexec.Error:
|
|
pass
|
|
else:
|
|
config._numfiles = len(getopenfiles(out))
|
|
|
|
#def pytest_report_header():
|
|
# return "pid: %s" % os.getpid()
|
|
|
|
def getopenfiles(out):
|
|
def isopen(line):
|
|
return ("REG" in line or "CHR" in line) and (
|
|
"deleted" not in line and 'mem' not in line and "txt" not in line)
|
|
return [x for x in out.split("\n") if isopen(x)]
|
|
|
|
def check_open_files(config):
|
|
out2 = py.process.cmdexec("lsof -p %d" % pid)
|
|
lines2 = getopenfiles(out2)
|
|
if len(lines2) > config._numfiles + 1:
|
|
error = []
|
|
error.append("***** %s FD leackage detected" %
|
|
(len(lines2)-config._numfiles))
|
|
error.extend(lines2)
|
|
error.append(error[0])
|
|
# update numfile so that the overall test run continuess
|
|
config._numfiles = len(lines2)
|
|
raise AssertionError("\n".join(error))
|
|
|
|
@pytest.mark.tryfirst # XXX rather do item.addfinalizer
|
|
def pytest_runtest_setup(item):
|
|
item._oldir = py.path.local()
|
|
|
|
def pytest_runtest_teardown(item, __multicall__):
|
|
item._oldir.chdir()
|
|
if hasattr(item.config, '_numfiles'):
|
|
x = __multicall__.execute()
|
|
check_open_files(item.config)
|
|
return x
|
|
|
|
def pytest_generate_tests(metafunc):
|
|
multi = getattr(metafunc.function, 'multi', None)
|
|
if multi is not None:
|
|
assert len(multi.kwargs) == 1
|
|
for name, l in multi.kwargs.items():
|
|
for val in l:
|
|
metafunc.addcall(funcargs={name: val})
|
|
elif 'anypython' in metafunc.fixturenames:
|
|
for name in ('python2.4', 'python2.5', 'python2.6',
|
|
'python2.7', 'python3.1', 'pypy', 'jython'):
|
|
metafunc.addcall(id=name, param=name)
|
|
|
|
# XXX copied from execnet's conftest.py - needs to be merged
|
|
winpymap = {
|
|
'python2.7': r'C:\Python27\python.exe',
|
|
'python2.6': r'C:\Python26\python.exe',
|
|
'python2.5': r'C:\Python25\python.exe',
|
|
'python2.4': r'C:\Python24\python.exe',
|
|
'python3.1': r'C:\Python31\python.exe',
|
|
}
|
|
|
|
def getexecutable(name, cache={}):
|
|
try:
|
|
return cache[name]
|
|
except KeyError:
|
|
executable = py.path.local.sysfind(name)
|
|
if executable:
|
|
if name == "jython":
|
|
import subprocess
|
|
popen = subprocess.Popen([str(executable), "--version"],
|
|
universal_newlines=True, stderr=subprocess.PIPE)
|
|
out, err = popen.communicate()
|
|
if not err or "2.5" not in err:
|
|
executable = None
|
|
if "2.5.2" in err:
|
|
executable = None # http://bugs.jython.org/issue1790
|
|
cache[name] = executable
|
|
return executable
|
|
|
|
def pytest_funcarg__anypython(request):
|
|
name = request.param
|
|
executable = getexecutable(name)
|
|
if executable is None:
|
|
if sys.platform == "win32":
|
|
executable = winpymap.get(name, None)
|
|
if executable:
|
|
executable = py.path.local(executable)
|
|
if executable.check():
|
|
return executable
|
|
pytest.skip("no suitable %s found" % (name,))
|
|
return executable
|