2008-08-16 23:26:59 +08:00
|
|
|
import py, sys, os
|
|
|
|
|
|
|
|
def setup_module(mod):
|
|
|
|
if not hasattr(os, 'fork'):
|
|
|
|
py.test.skip("forkedfunc requires os.fork")
|
|
|
|
mod.tmpdir = py.test.ensuretemp(mod.__file__)
|
|
|
|
|
|
|
|
def test_waitfinish_removes_tempdir():
|
2008-08-21 20:12:20 +08:00
|
|
|
ff = py.process.ForkedFunc(boxf1)
|
2008-08-16 23:26:59 +08:00
|
|
|
assert ff.tempdir.check()
|
|
|
|
ff.waitfinish()
|
|
|
|
assert not ff.tempdir.check()
|
|
|
|
|
2009-07-31 20:22:02 +08:00
|
|
|
def test_tempdir_gets_gc_collected(monkeypatch):
|
|
|
|
monkeypatch.setattr(os, 'fork', lambda: os.getpid())
|
2008-08-21 20:12:20 +08:00
|
|
|
ff = py.process.ForkedFunc(boxf1)
|
2008-08-16 23:26:59 +08:00
|
|
|
assert ff.tempdir.check()
|
|
|
|
ff.__del__()
|
|
|
|
assert not ff.tempdir.check()
|
|
|
|
|
|
|
|
def test_basic_forkedfunc():
|
2008-08-21 20:12:20 +08:00
|
|
|
result = py.process.ForkedFunc(boxf1).waitfinish()
|
2008-08-16 23:26:59 +08:00
|
|
|
assert result.out == "some out\n"
|
|
|
|
assert result.err == "some err\n"
|
|
|
|
assert result.exitstatus == 0
|
|
|
|
assert result.signal == 0
|
|
|
|
assert result.retval == 1
|
|
|
|
|
|
|
|
def test_exitstatus():
|
|
|
|
def func():
|
|
|
|
os._exit(4)
|
2008-08-21 20:12:20 +08:00
|
|
|
result = py.process.ForkedFunc(func).waitfinish()
|
2008-08-16 23:26:59 +08:00
|
|
|
assert result.exitstatus == 4
|
|
|
|
assert result.signal == 0
|
|
|
|
assert not result.out
|
|
|
|
assert not result.err
|
|
|
|
|
|
|
|
def test_execption_in_func():
|
|
|
|
def fun():
|
|
|
|
raise ValueError(42)
|
2008-08-21 20:12:20 +08:00
|
|
|
ff = py.process.ForkedFunc(fun)
|
2008-08-16 23:26:59 +08:00
|
|
|
result = ff.waitfinish()
|
|
|
|
assert result.exitstatus == ff.EXITSTATUS_EXCEPTION
|
|
|
|
assert result.err.find("ValueError: 42") != -1
|
|
|
|
assert result.signal == 0
|
|
|
|
assert not result.retval
|
|
|
|
|
|
|
|
def test_forkedfunc_on_fds():
|
2008-08-21 20:12:20 +08:00
|
|
|
result = py.process.ForkedFunc(boxf2).waitfinish()
|
2008-08-16 23:26:59 +08:00
|
|
|
assert result.out == "someout"
|
|
|
|
assert result.err == "someerr"
|
|
|
|
assert result.exitstatus == 0
|
|
|
|
assert result.signal == 0
|
|
|
|
assert result.retval == 2
|
|
|
|
|
|
|
|
def test_forkedfunc_signal():
|
2008-08-21 20:12:20 +08:00
|
|
|
result = py.process.ForkedFunc(boxseg).waitfinish()
|
2008-08-16 23:26:59 +08:00
|
|
|
assert result.retval is None
|
|
|
|
if py.std.sys.version_info < (2,4):
|
|
|
|
py.test.skip("signal detection does not work with python prior 2.4")
|
|
|
|
assert result.signal == 11
|
|
|
|
|
|
|
|
def test_forkedfunc_huge_data():
|
2008-08-21 20:12:20 +08:00
|
|
|
result = py.process.ForkedFunc(boxhuge).waitfinish()
|
2008-08-16 23:26:59 +08:00
|
|
|
assert result.out
|
|
|
|
assert result.exitstatus == 0
|
|
|
|
assert result.signal == 0
|
|
|
|
assert result.retval == 3
|
|
|
|
|
|
|
|
def test_box_seq():
|
|
|
|
# we run many boxes with huge data, just one after another
|
2009-08-29 22:40:03 +08:00
|
|
|
for i in range(50):
|
2008-08-21 20:12:20 +08:00
|
|
|
result = py.process.ForkedFunc(boxhuge).waitfinish()
|
2008-08-16 23:26:59 +08:00
|
|
|
assert result.out
|
|
|
|
assert result.exitstatus == 0
|
|
|
|
assert result.signal == 0
|
|
|
|
assert result.retval == 3
|
|
|
|
|
|
|
|
def test_box_in_a_box():
|
|
|
|
def boxfun():
|
2008-08-21 20:12:20 +08:00
|
|
|
result = py.process.ForkedFunc(boxf2).waitfinish()
|
2009-08-29 22:40:03 +08:00
|
|
|
print (result.out)
|
|
|
|
sys.stderr.write(result.err + "\n")
|
2008-08-16 23:26:59 +08:00
|
|
|
return result.retval
|
|
|
|
|
2008-08-21 20:12:20 +08:00
|
|
|
result = py.process.ForkedFunc(boxfun).waitfinish()
|
2008-08-16 23:26:59 +08:00
|
|
|
assert result.out == "someout\n"
|
|
|
|
assert result.err == "someerr\n"
|
|
|
|
assert result.exitstatus == 0
|
|
|
|
assert result.signal == 0
|
|
|
|
assert result.retval == 2
|
|
|
|
|
|
|
|
def test_kill_func_forked():
|
|
|
|
class A:
|
|
|
|
pass
|
|
|
|
info = A()
|
|
|
|
import time
|
|
|
|
|
|
|
|
def box_fun():
|
|
|
|
time.sleep(10) # we don't want to last forever here
|
|
|
|
|
2008-08-21 20:12:20 +08:00
|
|
|
ff = py.process.ForkedFunc(box_fun)
|
2008-08-16 23:26:59 +08:00
|
|
|
os.kill(ff.pid, 15)
|
|
|
|
result = ff.waitfinish()
|
|
|
|
if py.std.sys.version_info < (2,4):
|
|
|
|
py.test.skip("signal detection does not work with python prior 2.4")
|
|
|
|
assert result.signal == 15
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# ======================================================================
|
|
|
|
# examples
|
|
|
|
# ======================================================================
|
|
|
|
#
|
|
|
|
|
|
|
|
def boxf1():
|
2009-08-29 22:40:03 +08:00
|
|
|
sys.stdout.write("some out\n")
|
|
|
|
sys.stderr.write("some err\n")
|
2008-08-16 23:26:59 +08:00
|
|
|
return 1
|
|
|
|
|
|
|
|
def boxf2():
|
2009-08-29 22:40:03 +08:00
|
|
|
os.write(1, "someout".encode('ascii'))
|
|
|
|
os.write(2, "someerr".encode('ascii'))
|
2008-08-16 23:26:59 +08:00
|
|
|
return 2
|
|
|
|
|
|
|
|
def boxseg():
|
|
|
|
os.kill(os.getpid(), 11)
|
|
|
|
|
|
|
|
def boxhuge():
|
2009-08-29 22:40:03 +08:00
|
|
|
s = " ".encode('ascii')
|
|
|
|
os.write(1, s * 10000)
|
|
|
|
os.write(2, s * 10000)
|
|
|
|
os.write(1, s * 10000)
|
2008-08-16 23:26:59 +08:00
|
|
|
|
2009-08-29 22:40:03 +08:00
|
|
|
os.write(1, s * 10000)
|
|
|
|
os.write(2, s * 10000)
|
|
|
|
os.write(2, s * 10000)
|
|
|
|
os.write(1, s * 10000)
|
2008-08-16 23:26:59 +08:00
|
|
|
return 3
|