2011-07-08 18:37:20 +00:00
|
|
|
import argparse
|
|
|
|
import logging
|
|
|
|
import os
|
|
|
|
import subprocess
|
|
|
|
import sys
|
|
|
|
import tempfile
|
|
|
|
import yaml
|
|
|
|
|
|
|
|
import beanstalkc
|
|
|
|
|
|
|
|
from teuthology import safepath
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
def connect(ctx):
|
|
|
|
host = ctx.teuthology_config['queue_host']
|
|
|
|
port = ctx.teuthology_config['queue_port']
|
|
|
|
return beanstalkc.Connection(host=host, port=port)
|
|
|
|
|
|
|
|
def worker():
|
|
|
|
parser = argparse.ArgumentParser(description="""
|
|
|
|
Grab jobs from a beanstalk queue and run the teuthology tests they
|
|
|
|
describe. One job is run at a time.
|
|
|
|
""")
|
|
|
|
parser.add_argument(
|
|
|
|
'-v', '--verbose',
|
|
|
|
action='store_true', default=None,
|
|
|
|
help='be more verbose',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--archive-dir',
|
|
|
|
metavar='DIR',
|
|
|
|
help='path under which to archive results',
|
|
|
|
required=True,
|
|
|
|
)
|
2011-08-03 22:28:46 +00:00
|
|
|
parser.add_argument(
|
|
|
|
'-l', '--log-dir',
|
|
|
|
help='path in which to store logs',
|
|
|
|
required=True,
|
|
|
|
)
|
2011-07-08 18:37:20 +00:00
|
|
|
|
|
|
|
ctx = parser.parse_args()
|
|
|
|
|
|
|
|
loglevel = logging.INFO
|
|
|
|
if ctx.verbose:
|
|
|
|
loglevel = logging.DEBUG
|
|
|
|
|
|
|
|
logging.basicConfig(
|
|
|
|
level=loglevel,
|
2011-08-03 22:28:46 +00:00
|
|
|
filename=os.path.join(ctx.log_dir, 'worker.{pid}'.format(pid=os.getpid())),
|
|
|
|
format='%(asctime)s.%(msecs)03d %(levelname)s:%(name)s:%(message)s',
|
|
|
|
datefmt='%Y-%m-%dT%H:%M:%S',
|
2011-07-08 18:37:20 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
if not os.path.isdir(ctx.archive_dir):
|
|
|
|
sys.exit("{prog}: archive directory must exist: {path}".format(
|
|
|
|
prog=os.path.basename(sys.argv[0]),
|
|
|
|
path=ctx.archive_dir,
|
|
|
|
))
|
|
|
|
|
|
|
|
from teuthology.misc import read_config
|
|
|
|
read_config(ctx)
|
|
|
|
|
|
|
|
beanstalk = connect(ctx)
|
|
|
|
beanstalk.watch('teuthology')
|
|
|
|
beanstalk.ignore('default')
|
|
|
|
|
|
|
|
while True:
|
|
|
|
job = beanstalk.reserve(timeout=60)
|
|
|
|
if job is None:
|
|
|
|
continue
|
|
|
|
|
|
|
|
# bury the job so it won't be re-run if it fails
|
|
|
|
job.bury()
|
2011-08-26 00:11:33 +00:00
|
|
|
log.debug('Reserved job %d', job.jid)
|
2011-08-26 00:09:03 +00:00
|
|
|
log.debug('Config is: %s', job.body)
|
|
|
|
job_config = yaml.safe_load(job.body)
|
|
|
|
safe_archive = safepath.munge(job_config['name'])
|
2011-07-08 18:37:20 +00:00
|
|
|
|
2011-08-26 00:11:33 +00:00
|
|
|
if job_config.get('last_in_suite', False):
|
|
|
|
log.debug('Generating coverage for %s', job_config['name'])
|
2011-08-29 19:42:45 +00:00
|
|
|
args = [
|
|
|
|
os.path.join(os.path.dirname(sys.argv[0]), 'teuthology-results'),
|
|
|
|
'--timeout',
|
2011-09-23 01:23:36 +00:00
|
|
|
str(job_config.get('results_timeout', 21600)),
|
2011-08-29 19:42:45 +00:00
|
|
|
'--email',
|
|
|
|
job_config['email'],
|
|
|
|
'--archive-dir',
|
|
|
|
os.path.join(ctx.archive_dir, safe_archive),
|
|
|
|
'--name',
|
|
|
|
job_config['name'],
|
|
|
|
]
|
|
|
|
subprocess.Popen(args=args)
|
2011-08-26 00:11:33 +00:00
|
|
|
else:
|
|
|
|
log.debug('Creating archive dir...')
|
|
|
|
safepath.makedirs(ctx.archive_dir, safe_archive)
|
|
|
|
archive_path = os.path.join(ctx.archive_dir, safe_archive, str(job.jid))
|
|
|
|
log.info('Running job %d', job.jid)
|
|
|
|
run_job(job_config, archive_path)
|
2011-10-04 19:32:58 +00:00
|
|
|
job.delete()
|
2011-07-08 18:37:20 +00:00
|
|
|
|
2011-08-26 00:09:03 +00:00
|
|
|
def run_job(job_config, archive_path):
|
2011-07-08 18:37:20 +00:00
|
|
|
arg = [
|
|
|
|
os.path.join(os.path.dirname(sys.argv[0]), 'teuthology'),
|
|
|
|
]
|
|
|
|
|
|
|
|
if job_config['verbose']:
|
|
|
|
arg.append('-v')
|
|
|
|
|
|
|
|
arg.extend([
|
|
|
|
'--lock',
|
|
|
|
'--block',
|
|
|
|
'--owner', job_config['owner'],
|
|
|
|
'--archive', archive_path,
|
|
|
|
])
|
|
|
|
if job_config['description'] is not None:
|
|
|
|
arg.extend(['--description', job_config['description']])
|
|
|
|
arg.append('--')
|
|
|
|
|
2012-08-08 21:44:47 +00:00
|
|
|
with tempfile.NamedTemporaryFile(
|
|
|
|
prefix='teuthology-worker.',
|
|
|
|
suffix='.tmp',
|
|
|
|
) as tmp:
|
2012-08-09 16:42:13 +00:00
|
|
|
yaml.safe_dump(data=job_config['config'], stream=tmp)
|
2012-08-09 16:42:35 +00:00
|
|
|
tmp.flush()
|
2012-08-08 21:45:49 +00:00
|
|
|
arg.append(tmp.name)
|
2012-08-08 21:48:21 +00:00
|
|
|
p = subprocess.Popen(
|
|
|
|
args=arg,
|
|
|
|
close_fds=True,
|
|
|
|
stdout=subprocess.PIPE,
|
|
|
|
stderr=subprocess.STDOUT,
|
|
|
|
)
|
|
|
|
child = logging.getLogger(__name__ + '.child')
|
|
|
|
for line in p.stdout:
|
|
|
|
child.info(': %s', line.rstrip('\n'))
|
|
|
|
p.wait()
|
|
|
|
if p.returncode != 0:
|
|
|
|
log.error('Child exited with code %s', e)
|
2012-08-08 21:44:47 +00:00
|
|
|
else:
|
|
|
|
log.info('Success!')
|