194 lines
6.3 KiB
Python
194 lines
6.3 KiB
Python
import sys, os
|
|
import py
|
|
import time
|
|
import thread, threading
|
|
from py.__.test.rsession.master import MasterNode
|
|
from py.__.test.rsession.slave import setup_slave
|
|
|
|
from py.__.test.rsession import repevent
|
|
|
|
class HostInfo(object):
|
|
""" Class trying to store all necessary attributes
|
|
for host
|
|
"""
|
|
_hostname2list = {}
|
|
localdest = None
|
|
|
|
def __init__(self, spec):
|
|
parts = spec.split(':', 1)
|
|
self.hostname = parts.pop(0)
|
|
if parts:
|
|
self.relpath = parts[0]
|
|
else:
|
|
self.relpath = "pytestcache-" + self.hostname
|
|
self.hostid = self._getuniqueid(self.hostname)
|
|
|
|
def _getuniqueid(self, hostname):
|
|
l = self._hostname2list.setdefault(hostname, [])
|
|
hostid = hostname + "_" * len(l)
|
|
l.append(hostid)
|
|
return hostid
|
|
|
|
def initgateway(self, python="python"):
|
|
assert not hasattr(self, 'gw')
|
|
if self.hostname == "localhost":
|
|
gw = py.execnet.PopenGateway(python=python)
|
|
else:
|
|
gw = py.execnet.SshGateway(self.hostname,
|
|
remotepython=python)
|
|
self.gw = gw
|
|
channel = gw.remote_exec(py.code.Source(gethomedir, """
|
|
import os
|
|
targetdir = %r
|
|
homedir = gethomedir()
|
|
if not os.path.isabs(targetdir):
|
|
targetdir = os.path.join(homedir, targetdir)
|
|
if not os.path.exists(targetdir):
|
|
os.makedirs(targetdir)
|
|
os.chdir(homedir)
|
|
channel.send(targetdir)
|
|
""" % self.relpath))
|
|
self.gw_remotepath = channel.receive()
|
|
#print "initialized", gw, "with remotepath", self.gw_remotepath
|
|
if self.hostname == "localhost":
|
|
self.localdest = py.path.local(self.gw_remotepath)
|
|
assert self.localdest.check(dir=1)
|
|
|
|
def __str__(self):
|
|
return "<HostInfo %s:%s>" % (self.hostname, self.relpath)
|
|
|
|
def __repr__(self):
|
|
return str(self)
|
|
|
|
def __hash__(self):
|
|
return hash(self.hostid)
|
|
|
|
def __eq__(self, other):
|
|
return self.hostid == other.hostid
|
|
|
|
def __ne__(self, other):
|
|
return not self == other
|
|
|
|
class HostRSync(py.execnet.RSync):
|
|
""" RSyncer that filters out common files
|
|
"""
|
|
def __init__(self, *args, **kwargs):
|
|
self._synced = {}
|
|
ignores= None
|
|
if 'ignores' in kwargs:
|
|
ignores = kwargs.pop('ignores')
|
|
self._ignores = ignores or []
|
|
super(HostRSync, self).__init__(delete=True)
|
|
|
|
def filter(self, path):
|
|
path = py.path.local(path)
|
|
if not path.ext in ('.pyc', '.pyo'):
|
|
if not path.basename.endswith('~'):
|
|
if path.check(dotfile=0):
|
|
for x in self._ignores:
|
|
if path == x:
|
|
break
|
|
else:
|
|
return True
|
|
|
|
def add_target_host(self, host, destrelpath=None, finishedcallback=None):
|
|
key = host.hostname, host.relpath
|
|
if key in self._synced:
|
|
if finishedcallback:
|
|
finishedcallback()
|
|
return False
|
|
self._synced[key] = True
|
|
# the follow attributes are set from host.initgateway()
|
|
gw = host.gw
|
|
remotepath = host.gw_remotepath
|
|
if destrelpath is not None:
|
|
remotepath = os.path.join(remotepath, destrelpath)
|
|
super(HostRSync, self).add_target(gw,
|
|
remotepath,
|
|
finishedcallback)
|
|
return True # added the target
|
|
|
|
class HostManager(object):
|
|
def __init__(self, config, hosts=None):
|
|
self.config = config
|
|
if hosts is None:
|
|
hosts = self.config.getvalue("dist_hosts")
|
|
hosts = [HostInfo(x) for x in hosts]
|
|
self.hosts = hosts
|
|
|
|
def prepare_gateways(self):
|
|
dist_remotepython = self.config.getvalue("dist_remotepython")
|
|
for host in self.hosts:
|
|
host.initgateway(python=dist_remotepython)
|
|
host.gw.host = host
|
|
|
|
def _send_rsync(self, root, reporter, ignores, make_callback=False):
|
|
rsync = HostRSync(ignores=ignores)
|
|
destrelpath = root.relto(self.config.topdir)
|
|
for host in self.hosts:
|
|
reporter(repevent.HostRSyncing(host))
|
|
if make_callback:
|
|
def donecallback():
|
|
reporter(repevent.HostReady(host))
|
|
else:
|
|
donecallback = None
|
|
rsync.add_target_host(host, destrelpath,
|
|
finishedcallback=donecallback)
|
|
rsync.send(root)
|
|
|
|
|
|
def init_rsync(self, reporter):
|
|
# send each rsync root
|
|
roots = self.config.getvalue_pathlist("dist_rsync_roots")
|
|
ignores = self.config.getvalue_pathlist("dist_rsync_ignore")
|
|
if roots is None:
|
|
roots = [self.config.topdir]
|
|
self.prepare_gateways()
|
|
for root in roots[:-1]:
|
|
self._send_rsync(root, reporter, ignores)
|
|
self._send_rsync(roots[-1], reporter, ignores, True)
|
|
|
|
def setup_hosts(self, reporter):
|
|
self.init_rsync(reporter)
|
|
nodes = []
|
|
for host in self.hosts:
|
|
if hasattr(host.gw, 'remote_exec'): # otherwise dummy for tests :/
|
|
ch = setup_slave(host, self.config)
|
|
nodes.append(MasterNode(ch, reporter))
|
|
return nodes
|
|
|
|
def teardown_hosts(self, reporter, channels, nodes,
|
|
waiter=lambda : time.sleep(.1), exitfirst=False):
|
|
for channel in channels:
|
|
channel.send(None)
|
|
|
|
clean = exitfirst
|
|
while not clean:
|
|
clean = True
|
|
for node in nodes:
|
|
if node.pending:
|
|
clean = False
|
|
waiter()
|
|
self.teardown_gateways(reporter, channels)
|
|
|
|
def kill_channels(self, channels):
|
|
for channel in channels:
|
|
channel.send(42)
|
|
|
|
def teardown_gateways(self, reporter, channels):
|
|
for channel in channels:
|
|
try:
|
|
repevent.wrapcall(reporter, channel.waitclose)
|
|
except KeyboardInterrupt, SystemExit:
|
|
raise
|
|
except:
|
|
pass
|
|
channel.gateway.exit()
|
|
|
|
def gethomedir():
|
|
import os
|
|
homedir = os.environ.get('HOME', '')
|
|
if not homedir:
|
|
homedir = os.environ.get('HOMEPATH', '.')
|
|
return homedir
|