test_ok1/py/test/terminal/remote.py

128 lines
4.5 KiB
Python

from __future__ import generators
import py
from py.__.test.terminal.out import getout
import sys
def checkpyfilechange(rootdir, statcache={}):
""" wait until project files are changed. """
def fil(p):
return p.ext in ('.py', '.c', '.h')
#fil = lambda x: x.check(fnmatch='*.py')
def rec(p):
return p.check(dotfile=0)
changed = False
for path in rootdir.visit(fil, rec):
oldstat = statcache.get(path, None)
try:
statcache[path] = curstat = path.stat()
except py.error.ENOENT:
if oldstat:
del statcache[path]
print "# WARN: race condition on", path
else:
if oldstat:
if oldstat.mtime != curstat.mtime or \
oldstat.size != curstat.size:
changed = True
print "# MODIFIED", path
else:
changed = True
return changed
def getfailureitems(failures):
l = []
for rootpath, names in failures:
root = py.path.local(rootpath)
if root.check(dir=1):
current = py.test.collect.Directory(root).Directory(root)
elif root.check(file=1):
current = py.test.collect.Module(root).Module(root)
# root is fspath of names[0] -> pop names[0]
# slicing works with empty lists
names = names[1:]
while names:
name = names.pop(0)
try:
current = current.join(name)
except NameError:
print "WARNING: could not find %s on %r" %(name, current)
break
else:
l.append(current)
return l
class RemoteTerminalSession(object):
def __init__(self, config, file=None):
self.config = config
self._setexecutable()
if file is None:
file = py.std.sys.stdout
self._file = file
self.out = getout(file)
def _setexecutable(self):
name = self.config.option.executable
if name is None:
executable = py.std.sys.executable
else:
executable = py.path.local.sysfind(name)
assert executable is not None, executable
self.executable = executable
def main(self):
rootdir = self.config.topdir
wasfailing = False
failures = []
while 1:
if self.config.option.looponfailing and (failures or not wasfailing):
while not checkpyfilechange(rootdir):
py.std.time.sleep(0.4)
wasfailing = len(failures)
failures = self.run_remote_session(failures)
if not self.config.option.looponfailing:
break
print "#" * 60
print "# looponfailing: mode: %d failures args" % len(failures)
for root, names in failures:
name = "/".join(names) # XXX
print "Failure at: %r" % (name,)
print "# watching py files below %s" % rootdir
print "# ", "^" * len(str(rootdir))
return failures
def run_remote_session(self, failures):
print "* opening PopenGateway: ", self.executable
gw = py.execnet.PopenGateway(self.executable)
channel = gw.remote_exec("""
from py.__.test.terminal.remote import slaverun_TerminalSession
slaverun_TerminalSession(channel)
""", stdout=self.out, stderr=self.out)
print "MASTER: triggered slave terminal session ->"
repr = self.config.make_repr(conftestnames=[])
channel.send((str(self.config.topdir), repr, failures))
print "MASTER: send start info"
try:
return channel.receive()
except channel.RemoteError, e:
print e
return []
def slaverun_TerminalSession(channel):
""" we run this on the other side. """
print "SLAVE: starting"
topdir, repr, failures = channel.receive()
print "SLAVE: received configuration"
config = py.test.config
config.initdirect(topdir, repr, failures)
config.option.looponfailing = False
config.option.usepdb = False
config.option.executable = None
session = config.initsession()
session.shouldclose = channel.isclosed
print "SLAVE: starting session.main()"
session.main()
failures = session.getitemoutcomepairs(py.test.Item.Failed)
failures = [config.get_collector_trail(item) for item,_ in failures]
channel.send(failures)