128 lines
4.5 KiB
Python
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)
|