2018-08-12 13:01:04 +00:00
|
|
|
|
|
|
|
from __future__ import absolute_import
|
|
|
|
from __future__ import print_function
|
|
|
|
|
|
|
|
import atexit
|
|
|
|
import os
|
|
|
|
import shlex
|
|
|
|
import shutil
|
2018-11-06 13:47:09 +00:00
|
|
|
import subprocess
|
|
|
|
import sys
|
2018-08-12 13:01:04 +00:00
|
|
|
import tempfile
|
|
|
|
|
2018-11-06 13:47:09 +00:00
|
|
|
try:
|
|
|
|
import urlparse
|
|
|
|
except ImportError:
|
|
|
|
import urllib.parse as urlparse
|
|
|
|
|
2019-01-19 02:02:09 +00:00
|
|
|
os.chdir(
|
|
|
|
os.path.join(
|
|
|
|
os.path.dirname(__file__),
|
|
|
|
'..'
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
2018-08-12 13:01:04 +00:00
|
|
|
|
|
|
|
#
|
|
|
|
# check_output() monkeypatch cutpasted from testlib.py
|
|
|
|
#
|
|
|
|
|
|
|
|
def subprocess__check_output(*popenargs, **kwargs):
|
|
|
|
# Missing from 2.6.
|
|
|
|
process = subprocess.Popen(stdout=subprocess.PIPE, *popenargs, **kwargs)
|
|
|
|
output, _ = process.communicate()
|
|
|
|
retcode = process.poll()
|
|
|
|
if retcode:
|
|
|
|
cmd = kwargs.get("args")
|
|
|
|
if cmd is None:
|
|
|
|
cmd = popenargs[0]
|
|
|
|
raise subprocess.CalledProcessError(retcode, cmd)
|
|
|
|
return output
|
|
|
|
|
|
|
|
if not hasattr(subprocess, 'check_output'):
|
|
|
|
subprocess.check_output = subprocess__check_output
|
|
|
|
|
2018-10-26 10:36:49 +00:00
|
|
|
|
|
|
|
# -----------------
|
|
|
|
|
|
|
|
# Force stdout FD 1 to be a pipe, so tools like pip don't spam progress bars.
|
|
|
|
|
2018-11-05 21:09:54 +00:00
|
|
|
if sys.platform.startswith('linux'):
|
|
|
|
proc = subprocess.Popen(
|
|
|
|
args=['stdbuf', '-oL', 'cat'],
|
|
|
|
stdin=subprocess.PIPE
|
|
|
|
)
|
2018-10-26 10:36:49 +00:00
|
|
|
|
2018-11-05 21:09:54 +00:00
|
|
|
os.dup2(proc.stdin.fileno(), 1)
|
|
|
|
os.dup2(proc.stdin.fileno(), 2)
|
2018-11-05 12:39:39 +00:00
|
|
|
|
2018-11-05 21:09:54 +00:00
|
|
|
def cleanup_travis_junk(stdout=sys.stdout, stderr=sys.stderr, proc=proc):
|
|
|
|
stdout.close()
|
|
|
|
stderr.close()
|
|
|
|
proc.terminate()
|
2018-11-05 12:39:39 +00:00
|
|
|
|
2018-11-05 21:09:54 +00:00
|
|
|
atexit.register(cleanup_travis_junk)
|
2018-10-26 10:36:49 +00:00
|
|
|
|
2018-08-12 13:01:04 +00:00
|
|
|
# -----------------
|
|
|
|
|
|
|
|
def _argv(s, *args):
|
|
|
|
if args:
|
|
|
|
s %= args
|
|
|
|
return shlex.split(s)
|
|
|
|
|
|
|
|
|
|
|
|
def run(s, *args, **kwargs):
|
2018-11-06 13:47:09 +00:00
|
|
|
argv = ['/usr/bin/time', '--'] + _argv(s, *args)
|
2018-08-12 13:01:04 +00:00
|
|
|
print('Running: %s' % (argv,))
|
2018-11-05 12:39:39 +00:00
|
|
|
ret = subprocess.check_call(argv, **kwargs)
|
|
|
|
print('Finished running: %s' % (argv,))
|
|
|
|
return ret
|
2018-08-12 13:01:04 +00:00
|
|
|
|
|
|
|
|
2018-11-06 13:47:09 +00:00
|
|
|
def run_batches(batches):
|
|
|
|
combine = lambda batch: 'set -x; ' + (' && '.join(
|
|
|
|
'( %s; )' % (cmd,)
|
|
|
|
for cmd in batch
|
|
|
|
))
|
|
|
|
|
|
|
|
procs = [
|
|
|
|
subprocess.Popen(combine(batch), shell=True)
|
|
|
|
for batch in batches
|
|
|
|
]
|
|
|
|
assert [proc.wait() for proc in procs] == [0] * len(procs)
|
|
|
|
|
|
|
|
|
2018-08-12 13:01:04 +00:00
|
|
|
def get_output(s, *args, **kwargs):
|
|
|
|
argv = _argv(s, *args)
|
|
|
|
print('Running: %s' % (argv,))
|
|
|
|
return subprocess.check_output(argv, **kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
def exists_in_path(progname):
|
|
|
|
return any(os.path.exists(os.path.join(dirname, progname))
|
|
|
|
for dirname in os.environ['PATH'].split(os.pathsep))
|
|
|
|
|
|
|
|
|
|
|
|
class TempDir(object):
|
|
|
|
def __init__(self):
|
|
|
|
self.path = tempfile.mkdtemp(prefix='mitogen_ci_lib')
|
|
|
|
atexit.register(self.destroy)
|
|
|
|
|
|
|
|
def destroy(self, rmtree=shutil.rmtree):
|
|
|
|
rmtree(self.path)
|
|
|
|
|
|
|
|
|
|
|
|
class Fold(object):
|
|
|
|
def __init__(self, name):
|
|
|
|
self.name = name
|
|
|
|
|
|
|
|
def __enter__(self):
|
|
|
|
print('travis_fold:start:%s' % (self.name))
|
|
|
|
|
|
|
|
def __exit__(self, _1, _2, _3):
|
|
|
|
print('')
|
|
|
|
print('travis_fold:end:%s' % (self.name))
|
|
|
|
|
|
|
|
|
|
|
|
os.environ.setdefault('ANSIBLE_STRATEGY',
|
|
|
|
os.environ.get('STRATEGY', 'mitogen_linear'))
|
|
|
|
ANSIBLE_VERSION = os.environ.get('VER', '2.6.2')
|
|
|
|
GIT_ROOT = os.path.abspath(os.path.join(os.path.dirname(__file__), '..'))
|
2018-11-06 13:47:09 +00:00
|
|
|
DISTRO = os.environ.get('DISTRO', 'debian')
|
2018-08-12 13:01:04 +00:00
|
|
|
DISTROS = os.environ.get('DISTROS', 'debian centos6 centos7').split()
|
2018-11-06 13:47:09 +00:00
|
|
|
TARGET_COUNT = int(os.environ.get('TARGET_COUNT', '2'))
|
|
|
|
BASE_PORT = 2200
|
2018-08-12 13:01:04 +00:00
|
|
|
TMP = TempDir().path
|
|
|
|
|
|
|
|
os.environ['PYTHONDONTWRITEBYTECODE'] = 'x'
|
|
|
|
os.environ['PYTHONPATH'] = '%s:%s' % (
|
|
|
|
os.environ.get('PYTHONPATH', ''),
|
|
|
|
GIT_ROOT
|
|
|
|
)
|
|
|
|
|
2018-11-05 21:09:54 +00:00
|
|
|
def get_docker_hostname():
|
2018-11-06 13:47:09 +00:00
|
|
|
url = os.environ.get('DOCKER_HOST')
|
|
|
|
if url in (None, 'http+docker://localunixsocket'):
|
|
|
|
return 'localhost'
|
|
|
|
|
|
|
|
parsed = urlparse.urlparse(url)
|
|
|
|
return parsed.netloc.partition(':')[0]
|
|
|
|
|
|
|
|
|
2018-11-07 00:33:55 +00:00
|
|
|
def image_for_distro(distro):
|
|
|
|
return 'mitogen/%s-test' % (distro.partition('-')[0],)
|
|
|
|
|
|
|
|
|
2018-11-06 13:47:09 +00:00
|
|
|
def make_containers():
|
|
|
|
docker_hostname = get_docker_hostname()
|
2018-11-07 00:41:47 +00:00
|
|
|
firstbit = lambda s: (s+'-').split('-')[0]
|
|
|
|
secondbit = lambda s: (s+'-').split('-')[1]
|
|
|
|
|
2018-11-06 13:47:09 +00:00
|
|
|
return [
|
|
|
|
{
|
2018-11-07 00:41:47 +00:00
|
|
|
"distro": firstbit(distro),
|
2018-11-06 13:47:09 +00:00
|
|
|
"name": "target-%s-%s" % (distro, i),
|
|
|
|
"hostname": docker_hostname,
|
|
|
|
"port": BASE_PORT + i,
|
2018-11-07 00:41:47 +00:00
|
|
|
"python_path": (
|
|
|
|
'/usr/bin/python3'
|
|
|
|
if secondbit(distro) == 'py3'
|
|
|
|
else '/usr/bin/python'
|
|
|
|
)
|
2018-11-06 13:47:09 +00:00
|
|
|
}
|
|
|
|
for i, distro in enumerate(DISTROS, 1)
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
def start_containers(containers):
|
|
|
|
run_batches([
|
|
|
|
[
|
|
|
|
"docker rm -f %(name)s || true" % container,
|
|
|
|
"docker run "
|
|
|
|
"--rm "
|
|
|
|
"--detach "
|
|
|
|
"--publish 0.0.0.0:%(port)s:22/tcp "
|
|
|
|
"--hostname=%(name)s "
|
|
|
|
"--name=%(name)s "
|
|
|
|
"mitogen/%(distro)s-test "
|
|
|
|
% container
|
|
|
|
]
|
|
|
|
for container in containers
|
|
|
|
])
|
|
|
|
return containers
|
|
|
|
|
2018-08-12 13:01:04 +00:00
|
|
|
|
2018-11-07 00:40:37 +00:00
|
|
|
def dump_file(path):
|
|
|
|
print()
|
|
|
|
print('--- %s ---' % (path,))
|
|
|
|
print()
|
|
|
|
with open(path, 'r') as fp:
|
|
|
|
print(fp.read().rstrip())
|
|
|
|
print('---')
|
|
|
|
print()
|
|
|
|
|
|
|
|
|
2018-08-12 13:01:04 +00:00
|
|
|
# SSH passes these through to the container when run interactively, causing
|
|
|
|
# stdout to get messed up with libc warnings.
|
|
|
|
os.environ.pop('LANG', None)
|
|
|
|
os.environ.pop('LC_ALL', None)
|