155 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Python
		
	
	
	
			
		
		
	
	
			155 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Python
		
	
	
	
| """
 | |
|     LooponfailingSession and Helpers. 
 | |
| 
 | |
|     NOTE that one really has to avoid loading and depending on 
 | |
|     application modules within the controlling process 
 | |
|     (the one that starts repeatedly test processes)
 | |
|     otherwise changes to source code can crash 
 | |
|     the controlling process which should never happen. 
 | |
| """
 | |
| 
 | |
| from __future__ import generators
 | |
| import py
 | |
| from py.__.test.session import Session
 | |
| from py.__.test.dist.mypickle import PickleChannel
 | |
| from py.__.test import event
 | |
| from py.__.test.looponfail import util
 | |
| 
 | |
| class LooponfailingSession(Session):
 | |
|     def __init__(self, config):
 | |
|         super(LooponfailingSession, self).__init__(config=config)
 | |
|         self.rootdirs = [self.config.topdir] # xxx dist_rsync_roots? 
 | |
|         self.statrecorder = util.StatRecorder(self.rootdirs) 
 | |
|         self.remotecontrol = RemoteControl(self.config)
 | |
|         self.out = py.io.TerminalWriter()
 | |
| 
 | |
|     def main(self, initialitems=None):
 | |
|         try:
 | |
|             self.loopstate = loopstate = LoopState(initialitems)
 | |
|             self.remotecontrol.setup()
 | |
|             while 1:
 | |
|                 self.loop_once(loopstate)
 | |
|                 if not loopstate.colitems and loopstate.wasfailing:
 | |
|                     continue # the last failures passed, let's rerun all
 | |
|                 self.statrecorder.waitonchange(checkinterval=2.0) 
 | |
|         except KeyboardInterrupt:
 | |
|             print
 | |
| 
 | |
|     def loop_once(self, loopstate):
 | |
|         colitems = loopstate.colitems
 | |
|         loopstate.wasfailing = colitems and len(colitems)
 | |
|         loopstate.colitems = self.remotecontrol.runsession(colitems or ())
 | |
|         #ev = event.LooponfailingInfo(loopstate.failreports, self.rootdirs)
 | |
|         self.remotecontrol.setup()
 | |
| 
 | |
| class LoopState:
 | |
|     def __init__(self, colitems=None):
 | |
|         self.colitems = colitems
 | |
| 
 | |
| class RemoteControl(object):
 | |
|     def __init__(self, config):
 | |
|         self.config = config
 | |
| 
 | |
|     def trace(self, *args):
 | |
|         if self.config.option.debug:
 | |
|             msg = " ".join([str(x) for x in args])
 | |
|             print "RemoteControl:", msg 
 | |
| 
 | |
|     def initgateway(self):
 | |
|         return py.execnet.PopenGateway()
 | |
| 
 | |
|     def setup(self, out=None):
 | |
|         if out is None:
 | |
|             out = py.io.TerminalWriter()
 | |
|         if hasattr(self, 'gateway'):
 | |
|             raise ValueError("already have gateway %r" % self.gateway)
 | |
|         self.trace("setting up slave session")
 | |
|         old = self.config.topdir.chdir()
 | |
|         try:
 | |
|             self.gateway = self.initgateway()
 | |
|         finally:
 | |
|             old.chdir()
 | |
|         channel = self.gateway.remote_exec(source="""
 | |
|             from py.__.test.dist.mypickle import PickleChannel
 | |
|             from py.__.test.looponfail.remote import slave_runsession
 | |
|             channel = PickleChannel(channel)
 | |
|             config, fullwidth, hasmarkup = channel.receive()
 | |
|             slave_runsession(channel, config, fullwidth, hasmarkup) 
 | |
|         """, stdout=out, stderr=out)
 | |
|         channel = PickleChannel(channel)
 | |
|         channel.send((self.config, out.fullwidth, out.hasmarkup))
 | |
|         self.trace("set up of slave session complete")
 | |
|         self.channel = channel
 | |
| 
 | |
|     def ensure_teardown(self):
 | |
|         if hasattr(self, 'channel'):
 | |
|             if not self.channel.isclosed():
 | |
|                 self.trace("closing", self.channel)
 | |
|                 self.channel.close()
 | |
|             del self.channel
 | |
|         if hasattr(self, 'gateway'):
 | |
|             self.trace("exiting", self.gateway)
 | |
|             self.gateway.exit()
 | |
|             del self.gateway
 | |
| 
 | |
|     def runsession(self, colitems=()):
 | |
|         try:
 | |
|             self.trace("sending", colitems)
 | |
|             trails = colitems
 | |
|             self.channel.send(trails)
 | |
|             try:
 | |
|                 return self.channel.receive()
 | |
|             except self.channel.RemoteError, e:
 | |
|                 self.trace("ERROR", e)
 | |
|                 raise
 | |
|         finally:
 | |
|             self.ensure_teardown()
 | |
| 
 | |
| def slave_runsession(channel, config, fullwidth, hasmarkup):
 | |
|     """ we run this on the other side. """
 | |
|     if config.option.debug:
 | |
|         def DEBUG(*args): 
 | |
|             print " ".join(map(str, args))
 | |
|     else:
 | |
|         def DEBUG(*args): pass
 | |
| 
 | |
|     DEBUG("SLAVE: received configuration, using topdir:", config.topdir)
 | |
|     #config.option.session = None
 | |
|     config.option.looponfail = False 
 | |
|     config.option.usepdb = False 
 | |
|     trails = channel.receive()
 | |
|     config.pytestplugins.do_configure(config)
 | |
|     DEBUG("SLAVE: initsession()")
 | |
|     session = config.initsession()
 | |
|     # XXX configure the reporter object's terminal writer more directly
 | |
|     # XXX and write a test for this remote-terminal setting logic 
 | |
|     config.pytest_terminal_hasmarkup = hasmarkup
 | |
|     config.pytest_terminal_fullwidth = fullwidth
 | |
|     if trails:
 | |
|         colitems = []
 | |
|         for trail in trails:
 | |
|             try:
 | |
|                 colitem = py.test.collect.Collector._fromtrail(trail, config)
 | |
|             except AssertionError, e:  
 | |
|                 #XXX session.bus.notify of "test disappeared"
 | |
|                 continue 
 | |
|             colitems.append(colitem)
 | |
|     else:
 | |
|         colitems = None
 | |
|     session.shouldclose = channel.isclosed 
 | |
|    
 | |
|     class Failures(list):
 | |
|         def pyevent_itemtestreport(self, ev):
 | |
|             if ev.failed:
 | |
|                 self.append(ev)
 | |
|         pyevent_collectionreport = pyevent_itemtestreport
 | |
|         
 | |
|     failreports = Failures()
 | |
|     session.bus.register(failreports)
 | |
| 
 | |
|     DEBUG("SLAVE: starting session.main()")
 | |
|     session.main(colitems)
 | |
|     ev = event.LooponfailingInfo(list(failreports), [config.topdir])
 | |
|     session.bus.notify("looponfailinfo", ev)
 | |
|     channel.send([x.colitem._totrail() for x in failreports if x.failed])
 |