2013-08-23 06:22:04 +00:00
|
|
|
# this file is responsible for submitting tests into the queue
|
|
|
|
# by generating combinations of facets found in
|
|
|
|
# https://github.com/ceph/ceph-qa-suite.git
|
|
|
|
|
2011-08-26 00:11:33 +00:00
|
|
|
import copy
|
2014-06-19 17:41:51 +00:00
|
|
|
from datetime import datetime
|
2011-06-21 17:00:16 +00:00
|
|
|
import itertools
|
|
|
|
import logging
|
|
|
|
import os
|
2014-06-12 17:30:34 +00:00
|
|
|
import requests
|
2014-06-19 17:41:51 +00:00
|
|
|
import pwd
|
2011-06-21 17:00:16 +00:00
|
|
|
import subprocess
|
2014-06-19 17:41:51 +00:00
|
|
|
import smtplib
|
2011-06-21 17:00:16 +00:00
|
|
|
import sys
|
2011-08-26 00:11:33 +00:00
|
|
|
import yaml
|
2014-06-19 17:41:51 +00:00
|
|
|
from email.mime.text import MIMEText
|
|
|
|
from tempfile import NamedTemporaryFile
|
2011-08-26 00:11:33 +00:00
|
|
|
|
2013-10-11 22:10:57 +00:00
|
|
|
import teuthology
|
2014-06-25 20:59:38 +00:00
|
|
|
from . import lock
|
2014-07-10 01:02:47 +00:00
|
|
|
from .config import config, JobConfig
|
2014-06-30 16:10:31 +00:00
|
|
|
from .repo_utils import enforce_repo_state, BranchNotFoundError
|
2011-06-21 17:00:16 +00:00
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
2013-09-26 15:47:43 +00:00
|
|
|
|
2013-10-08 16:34:09 +00:00
|
|
|
def main(args):
|
2014-06-11 20:13:45 +00:00
|
|
|
verbose = args['--verbose']
|
2014-06-19 17:41:51 +00:00
|
|
|
if verbose:
|
|
|
|
teuthology.log.setLevel(logging.DEBUG)
|
2014-06-11 20:13:45 +00:00
|
|
|
dry_run = args['--dry-run']
|
2014-06-19 17:41:51 +00:00
|
|
|
|
|
|
|
base_yaml_paths = args['<config_yaml>']
|
2014-07-10 01:02:47 +00:00
|
|
|
suite = args['--suite'].replace('/', ':')
|
2014-06-19 17:41:51 +00:00
|
|
|
ceph_branch = args['--ceph']
|
|
|
|
kernel_branch = args['--kernel']
|
|
|
|
kernel_flavor = args['--flavor']
|
|
|
|
teuthology_branch = args['--teuthology-branch']
|
|
|
|
machine_type = args['--machine-type']
|
|
|
|
distro = args['--distro']
|
2014-07-14 20:30:31 +00:00
|
|
|
suite_branch = args['--suite-branch']
|
2014-07-01 18:21:45 +00:00
|
|
|
suite_dir = args['--suite-dir']
|
2014-06-19 17:41:51 +00:00
|
|
|
|
|
|
|
limit = int(args['--limit'])
|
2014-06-11 20:13:45 +00:00
|
|
|
priority = int(args['--priority'])
|
|
|
|
num = int(args['--num'])
|
|
|
|
owner = args['--owner']
|
|
|
|
email = args['--email']
|
2014-06-19 17:41:51 +00:00
|
|
|
if email:
|
|
|
|
config.results_email = email
|
2014-06-11 20:13:45 +00:00
|
|
|
timeout = args['--timeout']
|
2011-06-21 17:00:16 +00:00
|
|
|
|
2014-07-10 01:02:47 +00:00
|
|
|
name = make_run_name(suite, ceph_branch, kernel_branch, kernel_flavor,
|
2014-06-24 15:38:13 +00:00
|
|
|
machine_type)
|
2014-06-25 20:59:38 +00:00
|
|
|
|
2014-07-14 20:30:31 +00:00
|
|
|
job_config = create_initial_config(suite, suite_branch, ceph_branch,
|
|
|
|
teuthology_branch, kernel_branch,
|
|
|
|
kernel_flavor, distro, machine_type)
|
|
|
|
|
2014-07-01 18:21:45 +00:00
|
|
|
if suite_dir:
|
|
|
|
suite_repo_path = suite_dir
|
2014-07-01 17:45:10 +00:00
|
|
|
else:
|
2014-07-14 20:30:31 +00:00
|
|
|
suite_repo_path = fetch_suite_repo(job_config.suite_branch,
|
|
|
|
test_name=name)
|
2014-06-19 17:41:51 +00:00
|
|
|
|
2014-07-10 01:02:47 +00:00
|
|
|
job_config.name = name
|
|
|
|
job_config.priority = priority
|
2014-07-15 17:28:07 +00:00
|
|
|
if config.results_email:
|
|
|
|
job_config.email = config.results_email
|
2014-07-10 01:02:47 +00:00
|
|
|
if owner:
|
|
|
|
job_config.owner = owner
|
|
|
|
|
2014-06-23 22:53:03 +00:00
|
|
|
with NamedTemporaryFile(prefix='schedule_suite_',
|
|
|
|
delete=False) as base_yaml:
|
2014-07-10 01:02:47 +00:00
|
|
|
base_yaml.write(str(job_config))
|
2014-06-23 22:53:03 +00:00
|
|
|
base_yaml_path = base_yaml.name
|
|
|
|
base_yaml_paths.insert(0, base_yaml_path)
|
2014-07-10 01:02:47 +00:00
|
|
|
prepare_and_schedule(job_config=job_config,
|
2014-06-25 20:59:38 +00:00
|
|
|
suite_repo_path=suite_repo_path,
|
2014-06-25 20:24:19 +00:00
|
|
|
base_yaml_paths=base_yaml_paths,
|
|
|
|
limit=limit,
|
|
|
|
num=num,
|
|
|
|
timeout=timeout,
|
|
|
|
dry_run=dry_run,
|
|
|
|
verbose=verbose,
|
|
|
|
)
|
2014-06-23 22:53:03 +00:00
|
|
|
os.remove(base_yaml_path)
|
2014-06-19 17:41:51 +00:00
|
|
|
|
|
|
|
|
2014-06-24 15:38:13 +00:00
|
|
|
def make_run_name(suite, ceph_branch, kernel_branch, kernel_flavor,
|
|
|
|
machine_type, user=None, timestamp=None):
|
|
|
|
"""
|
|
|
|
Generate a run name based on the parameters. A run name looks like:
|
|
|
|
teuthology-2014-06-23_19:00:37-rados-dumpling-testing-basic-plana
|
|
|
|
"""
|
2014-06-19 17:41:51 +00:00
|
|
|
if not user:
|
|
|
|
user = pwd.getpwuid(os.getuid()).pw_name
|
|
|
|
# We assume timestamp is a datetime.datetime object
|
|
|
|
if not timestamp:
|
|
|
|
timestamp = datetime.now().strftime('%Y-%m-%d_%H:%M:%S')
|
|
|
|
|
|
|
|
worker = get_worker(machine_type)
|
|
|
|
return '-'.join(
|
|
|
|
[user, str(timestamp), suite, ceph_branch,
|
|
|
|
kernel_branch, kernel_flavor, worker]
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2014-06-25 20:59:38 +00:00
|
|
|
def fetch_suite_repo(branch, test_name):
|
|
|
|
"""
|
2014-06-25 21:10:37 +00:00
|
|
|
Fetch the suite repo (and also the teuthology repo) so that we can use it
|
|
|
|
to build jobs. Repos are stored in ~/src/.
|
2014-06-25 20:59:38 +00:00
|
|
|
|
2014-06-27 16:12:45 +00:00
|
|
|
The reason the teuthology repo is also fetched is that currently we use
|
|
|
|
subprocess to call teuthology-schedule to schedule jobs so we need to make
|
|
|
|
sure it is up-to-date. For that reason we always fetch the master branch
|
|
|
|
for test scheduling, regardless of what teuthology branch is requested for
|
|
|
|
testing.
|
|
|
|
|
2014-06-25 20:59:38 +00:00
|
|
|
:returns: The path to the repo on disk
|
|
|
|
"""
|
2014-07-10 07:19:48 +00:00
|
|
|
src_base_path = config.src_base_path
|
2014-06-25 20:59:38 +00:00
|
|
|
if not os.path.exists(src_base_path):
|
|
|
|
os.mkdir(src_base_path)
|
|
|
|
suite_repo_path = os.path.join(src_base_path,
|
|
|
|
'ceph-qa-suite_' + branch)
|
|
|
|
try:
|
2014-07-03 16:18:01 +00:00
|
|
|
# When a user is scheduling a test run from their own copy of
|
|
|
|
# teuthology, let's not wreak havoc on it.
|
|
|
|
if config.automated_scheduling:
|
|
|
|
enforce_repo_state(
|
|
|
|
repo_url=os.path.join(config.ceph_git_base_url,
|
|
|
|
'teuthology.git'),
|
|
|
|
dest_path=os.path.join(src_base_path, 'teuthology'),
|
|
|
|
branch='master',
|
|
|
|
remove_on_error=False,
|
|
|
|
)
|
2014-06-30 16:10:31 +00:00
|
|
|
enforce_repo_state(
|
2014-06-25 21:24:05 +00:00
|
|
|
repo_url=os.path.join(config.ceph_git_base_url,
|
|
|
|
'ceph-qa-suite.git'),
|
2014-06-25 20:59:38 +00:00
|
|
|
dest_path=suite_repo_path,
|
2014-07-03 00:27:09 +00:00
|
|
|
branch=branch,
|
|
|
|
)
|
2014-06-25 20:59:38 +00:00
|
|
|
except BranchNotFoundError as exc:
|
|
|
|
schedule_fail(message=str(exc), name=test_name)
|
|
|
|
return suite_repo_path
|
|
|
|
|
|
|
|
|
2014-07-14 20:30:31 +00:00
|
|
|
def create_initial_config(suite, suite_branch, ceph_branch, teuthology_branch,
|
2014-06-19 17:41:51 +00:00
|
|
|
kernel_branch, kernel_flavor, distro, machine_type):
|
2014-06-25 15:32:56 +00:00
|
|
|
"""
|
|
|
|
Put together the config file used as the basis for each job in the run.
|
|
|
|
Grabs hashes for the latest ceph, kernel and teuthology versions in the
|
|
|
|
branches specified and specifies them so we know exactly what we're
|
|
|
|
testing.
|
|
|
|
|
2014-07-14 20:30:31 +00:00
|
|
|
:returns: A JobConfig object
|
2014-06-25 15:32:56 +00:00
|
|
|
"""
|
2014-06-19 17:41:51 +00:00
|
|
|
# Put together a stanza specifying the kernel hash
|
|
|
|
if kernel_branch == 'distro':
|
|
|
|
kernel_hash = 'distro'
|
2014-06-25 16:02:59 +00:00
|
|
|
# Skip the stanza if the branch passed is '-'
|
|
|
|
elif kernel_branch == '-':
|
|
|
|
kernel_hash = None
|
2014-06-19 17:41:51 +00:00
|
|
|
else:
|
|
|
|
kernel_hash = get_hash('kernel', kernel_branch, kernel_flavor,
|
|
|
|
machine_type)
|
|
|
|
if not kernel_hash:
|
2014-06-25 20:59:38 +00:00
|
|
|
schedule_fail(message="Kernel branch '{branch}' not found".format(
|
2014-06-19 17:41:51 +00:00
|
|
|
branch=kernel_branch))
|
|
|
|
if kernel_hash:
|
|
|
|
log.info("kernel sha1: {hash}".format(hash=kernel_hash))
|
|
|
|
kernel_dict = dict(kernel=dict(kdb=True, sha1=kernel_hash))
|
2014-06-25 16:02:59 +00:00
|
|
|
else:
|
2014-07-09 17:46:01 +00:00
|
|
|
kernel_dict = dict()
|
2014-06-19 17:41:51 +00:00
|
|
|
|
|
|
|
# Get the ceph hash
|
|
|
|
ceph_hash = get_hash('ceph', ceph_branch, kernel_flavor, machine_type)
|
|
|
|
if not ceph_hash:
|
2014-07-14 20:30:31 +00:00
|
|
|
exc = BranchNotFoundError(ceph_branch, 'ceph.git')
|
|
|
|
schedule_fail(message=str(exc))
|
2014-06-19 17:41:51 +00:00
|
|
|
log.info("ceph sha1: {hash}".format(hash=ceph_hash))
|
|
|
|
|
|
|
|
# Get the ceph package version
|
|
|
|
ceph_version = package_version_for_hash(ceph_hash, kernel_flavor,
|
2014-07-29 20:43:41 +00:00
|
|
|
distro, machine_type)
|
2014-06-19 17:41:51 +00:00
|
|
|
if not ceph_version:
|
|
|
|
schedule_fail("Packages for ceph version '{ver}' not found".format(
|
|
|
|
ver=ceph_version))
|
|
|
|
log.info("ceph version: {ver}".format(ver=ceph_version))
|
|
|
|
|
|
|
|
# Decide what branch of s3-tests to use
|
|
|
|
if get_branch_info('s3-tests', ceph_branch):
|
|
|
|
s3_branch = ceph_branch
|
|
|
|
else:
|
|
|
|
log.info("branch {0} not in s3-tests.git; will use master for"
|
2014-07-14 20:30:31 +00:00
|
|
|
" s3-tests".format(ceph_branch))
|
2014-06-19 17:41:51 +00:00
|
|
|
s3_branch = 'master'
|
|
|
|
log.info("s3-tests branch: %s", s3_branch)
|
|
|
|
|
2014-07-14 20:30:31 +00:00
|
|
|
if teuthology_branch:
|
|
|
|
if not get_branch_info('teuthology', teuthology_branch):
|
|
|
|
exc = BranchNotFoundError(teuthology_branch, 'teuthology.git')
|
|
|
|
raise schedule_fail(message=str(exc))
|
|
|
|
else:
|
2014-06-19 17:41:51 +00:00
|
|
|
# Decide what branch of teuthology to use
|
|
|
|
if get_branch_info('teuthology', ceph_branch):
|
|
|
|
teuthology_branch = ceph_branch
|
|
|
|
else:
|
|
|
|
log.info("branch {0} not in teuthology.git; will use master for"
|
2014-07-14 20:30:31 +00:00
|
|
|
" teuthology".format(ceph_branch))
|
2014-06-19 17:41:51 +00:00
|
|
|
teuthology_branch = 'master'
|
|
|
|
log.info("teuthology branch: %s", teuthology_branch)
|
|
|
|
|
2014-07-14 20:30:31 +00:00
|
|
|
if not suite_branch:
|
|
|
|
# Decide what branch of ceph-qa-suite to use
|
|
|
|
if get_branch_info('ceph-qa-suite', ceph_branch):
|
|
|
|
suite_branch = ceph_branch
|
|
|
|
else:
|
|
|
|
log.info("branch {0} not in ceph-qa-suite.git; will use master for"
|
|
|
|
" ceph-qa-suite".format(ceph_branch))
|
|
|
|
suite_branch = 'master'
|
|
|
|
log.info("ceph-qa-suite branch: %s", suite_branch)
|
|
|
|
|
2014-06-19 17:41:51 +00:00
|
|
|
config_input = dict(
|
2014-07-10 01:02:47 +00:00
|
|
|
suite=suite,
|
2014-07-14 20:30:31 +00:00
|
|
|
suite_branch=suite_branch,
|
2014-06-19 17:41:51 +00:00
|
|
|
ceph_branch=ceph_branch,
|
|
|
|
ceph_hash=ceph_hash,
|
|
|
|
teuthology_branch=teuthology_branch,
|
|
|
|
machine_type=machine_type,
|
|
|
|
distro=distro,
|
|
|
|
s3_branch=s3_branch,
|
|
|
|
)
|
2014-07-10 01:02:47 +00:00
|
|
|
conf_dict = substitute_placeholders(dict_templ, config_input)
|
|
|
|
conf_dict.update(kernel_dict)
|
|
|
|
job_config = JobConfig.from_dict(conf_dict)
|
2014-07-09 17:46:01 +00:00
|
|
|
return job_config
|
2014-06-16 18:26:50 +00:00
|
|
|
|
|
|
|
|
2014-07-10 01:02:47 +00:00
|
|
|
def prepare_and_schedule(job_config, suite_repo_path, base_yaml_paths, limit,
|
|
|
|
num, timeout, dry_run, verbose):
|
2014-06-25 15:32:56 +00:00
|
|
|
"""
|
|
|
|
Puts together some "base arguments" with which to execute
|
|
|
|
teuthology-schedule for each job, then passes them and other parameters to
|
|
|
|
schedule_suite(). Finally, schedules a "last-in-suite" job that sends an
|
2014-07-03 00:39:47 +00:00
|
|
|
email to the specified address (if one is configured).
|
2014-06-25 15:32:56 +00:00
|
|
|
"""
|
2014-07-10 01:02:47 +00:00
|
|
|
arch = get_arch(job_config.machine_type)
|
2014-06-12 17:43:49 +00:00
|
|
|
|
2014-06-10 20:50:59 +00:00
|
|
|
base_args = [
|
2011-08-26 00:11:33 +00:00
|
|
|
os.path.join(os.path.dirname(sys.argv[0]), 'teuthology-schedule'),
|
2014-07-10 01:02:47 +00:00
|
|
|
'--name', job_config.name,
|
2014-06-10 20:50:59 +00:00
|
|
|
'--num', str(num),
|
2014-07-10 01:02:47 +00:00
|
|
|
'--worker', get_worker(job_config.machine_type),
|
2013-10-08 22:10:01 +00:00
|
|
|
]
|
2014-07-10 01:02:47 +00:00
|
|
|
if job_config.priority:
|
|
|
|
base_args.extend(['--priority', str(job_config.priority)])
|
2014-06-10 20:50:59 +00:00
|
|
|
if verbose:
|
|
|
|
base_args.append('-v')
|
2014-07-10 01:02:47 +00:00
|
|
|
if job_config.owner:
|
|
|
|
base_args.extend(['--owner', job_config.owner])
|
2011-08-26 00:11:33 +00:00
|
|
|
|
2014-07-10 01:02:47 +00:00
|
|
|
suite_path = os.path.join(suite_repo_path, 'suites',
|
|
|
|
job_config.suite.replace(':', '/'))
|
2014-06-06 21:12:35 +00:00
|
|
|
|
2014-07-22 21:47:53 +00:00
|
|
|
# Make sure the yaml paths are actually valid
|
|
|
|
for yaml_path in base_yaml_paths:
|
|
|
|
full_yaml_path = os.path.join(suite_repo_path, yaml_path)
|
|
|
|
if not os.path.exists(full_yaml_path):
|
|
|
|
raise IOError("File not found: " + full_yaml_path)
|
|
|
|
|
2014-06-13 19:48:18 +00:00
|
|
|
num_jobs = schedule_suite(
|
2014-07-10 01:02:47 +00:00
|
|
|
job_config=job_config,
|
2014-06-13 19:48:18 +00:00
|
|
|
path=suite_path,
|
|
|
|
base_yamls=base_yaml_paths,
|
|
|
|
base_args=base_args,
|
|
|
|
arch=arch,
|
|
|
|
limit=limit,
|
|
|
|
dry_run=dry_run,
|
|
|
|
)
|
2013-08-22 21:39:56 +00:00
|
|
|
|
2014-07-10 01:02:47 +00:00
|
|
|
if job_config.email and num_jobs:
|
2014-06-10 20:50:59 +00:00
|
|
|
arg = copy.deepcopy(base_args)
|
|
|
|
arg.append('--last-in-suite')
|
2014-07-10 01:02:47 +00:00
|
|
|
arg.extend(['--email', job_config.email])
|
2014-06-10 20:50:59 +00:00
|
|
|
if timeout:
|
|
|
|
arg.extend(['--timeout', timeout])
|
|
|
|
if dry_run:
|
|
|
|
log.info('dry-run: %s' % ' '.join(arg))
|
|
|
|
else:
|
|
|
|
subprocess.check_call(
|
|
|
|
args=arg,
|
2013-10-08 22:10:01 +00:00
|
|
|
)
|
2013-08-22 21:39:56 +00:00
|
|
|
|
2013-08-23 06:29:27 +00:00
|
|
|
|
2014-06-25 20:59:38 +00:00
|
|
|
def schedule_fail(message, name=''):
|
2014-06-25 15:32:56 +00:00
|
|
|
"""
|
|
|
|
If an email address has been specified anywhere, send an alert there. Then
|
|
|
|
raise a ScheduleFailError.
|
|
|
|
"""
|
2014-06-19 17:41:51 +00:00
|
|
|
email = config.results_email
|
2014-06-19 19:06:21 +00:00
|
|
|
if email:
|
|
|
|
subject = "Failed to schedule {name}".format(name=name)
|
|
|
|
msg = MIMEText(message)
|
|
|
|
msg['Subject'] = subject
|
|
|
|
msg['From'] = config.results_sending_email
|
|
|
|
msg['To'] = email
|
|
|
|
smtp = smtplib.SMTP('localhost')
|
|
|
|
smtp.sendmail(msg['From'], [msg['To']], msg.as_string())
|
|
|
|
smtp.quit()
|
2014-06-19 17:41:51 +00:00
|
|
|
raise ScheduleFailError(message, name)
|
|
|
|
|
|
|
|
|
|
|
|
class ScheduleFailError(RuntimeError):
|
|
|
|
def __init__(self, message, name=None):
|
|
|
|
self.message = message
|
|
|
|
self.name = name
|
|
|
|
|
|
|
|
def __str__(self):
|
2014-07-14 20:30:31 +00:00
|
|
|
return "Job scheduling {name} failed: {msg}".format(
|
2014-06-19 17:41:51 +00:00
|
|
|
name=self.name,
|
|
|
|
msg=self.message,
|
|
|
|
).replace(' ', ' ')
|
|
|
|
|
|
|
|
|
|
|
|
def get_worker(machine_type):
|
2014-06-24 15:38:13 +00:00
|
|
|
"""
|
|
|
|
Map a given machine_type to a beanstalkd worker. If machine_type mentions
|
|
|
|
multiple machine types - e.g. 'plana,mira', then this returns 'multi'.
|
|
|
|
Otherwise it returns what was passed.
|
|
|
|
"""
|
2014-06-19 17:41:51 +00:00
|
|
|
if ',' in machine_type:
|
|
|
|
return 'multi'
|
|
|
|
else:
|
|
|
|
return machine_type
|
|
|
|
|
|
|
|
|
|
|
|
def get_hash(project='ceph', branch='master', flavor='basic',
|
2014-07-16 18:07:55 +00:00
|
|
|
machine_type='plana', distro='ubuntu'):
|
2014-06-24 15:38:13 +00:00
|
|
|
"""
|
|
|
|
Find the hash representing the head of the project's repository via
|
|
|
|
querying a gitbuilder repo.
|
|
|
|
|
|
|
|
Will return None in the case of a 404 or any other HTTP error.
|
|
|
|
"""
|
|
|
|
# Alternate method for github-hosted projects - left here for informational
|
|
|
|
# purposes
|
2014-06-25 20:24:19 +00:00
|
|
|
# resp = requests.get(
|
|
|
|
# 'https://api.github.com/repos/ceph/ceph/git/refs/heads/master')
|
|
|
|
# hash = .json()['object']['sha']
|
2014-06-19 17:41:51 +00:00
|
|
|
(arch, release, pkg_type) = get_distro_defaults(distro, machine_type)
|
|
|
|
base_url = get_gitbuilder_url(project, release, pkg_type, arch, flavor)
|
|
|
|
url = os.path.join(base_url, 'ref', branch, 'sha1')
|
|
|
|
resp = requests.get(url)
|
|
|
|
if not resp.ok:
|
|
|
|
return None
|
|
|
|
return str(resp.text.strip())
|
|
|
|
|
|
|
|
|
|
|
|
def get_distro_defaults(distro, machine_type):
|
|
|
|
"""
|
|
|
|
Given a distro (e.g. 'ubuntu') and machine type, return:
|
|
|
|
(arch, release, pkg_type)
|
|
|
|
|
|
|
|
This is mainly used to default to:
|
|
|
|
('x86_64', 'precise', 'deb') when passed 'ubuntu' and 'plana'
|
|
|
|
And ('armv7l', 'saucy', 'deb') when passed 'ubuntu' and 'saya'
|
|
|
|
And ('x86_64', 'centos6', 'rpm') when passed anything non-ubuntu
|
|
|
|
"""
|
|
|
|
if distro == 'ubuntu':
|
|
|
|
if machine_type == 'saya':
|
|
|
|
arch = 'armv7l'
|
|
|
|
release = 'saucy'
|
|
|
|
pkg_type = 'deb'
|
|
|
|
else:
|
|
|
|
arch = 'x86_64'
|
|
|
|
release = 'precise'
|
|
|
|
pkg_type = 'deb'
|
|
|
|
else:
|
|
|
|
arch = 'x86_64'
|
|
|
|
release = 'centos6'
|
|
|
|
pkg_type = 'rpm'
|
2014-06-24 15:38:13 +00:00
|
|
|
log.debug(
|
|
|
|
"Defaults for machine_type %s: arch=%s, release=%s, pkg_type=%s)",
|
|
|
|
machine_type, arch, release, pkg_type)
|
2014-06-19 17:41:51 +00:00
|
|
|
return (
|
|
|
|
arch,
|
|
|
|
release,
|
|
|
|
pkg_type,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def get_gitbuilder_url(project, distro, pkg_type, arch, kernel_flavor):
|
2014-06-12 17:30:34 +00:00
|
|
|
"""
|
|
|
|
Return a base URL like:
|
|
|
|
http://gitbuilder.ceph.com/ceph-deb-squeeze-x86_64-basic/
|
|
|
|
|
2014-06-19 17:41:51 +00:00
|
|
|
:param project: 'ceph' or 'kernel'
|
2014-06-12 17:30:34 +00:00
|
|
|
:param distro: A distro-ish string like 'trusty' or 'fedora20'
|
|
|
|
:param pkg_type: Probably 'rpm' or 'deb'
|
|
|
|
:param arch: A string like 'x86_64'
|
|
|
|
:param kernel_flavor: A string like 'basic'
|
|
|
|
"""
|
2014-06-19 17:41:51 +00:00
|
|
|
templ = 'http://gitbuilder.ceph.com/{proj}-{pkg}-{distro}-{arch}-{flav}/'
|
|
|
|
return templ.format(proj=project, pkg=pkg_type, distro=distro, arch=arch,
|
2014-06-12 17:30:34 +00:00
|
|
|
flav=kernel_flavor)
|
|
|
|
|
|
|
|
|
2014-06-19 17:41:51 +00:00
|
|
|
def package_version_for_hash(hash, kernel_flavor='basic',
|
|
|
|
distro='ubuntu', machine_type='plana'):
|
2014-06-25 15:32:56 +00:00
|
|
|
"""
|
|
|
|
Does what it says on the tin. Uses gitbuilder repos.
|
|
|
|
|
|
|
|
:returns: a string.
|
|
|
|
"""
|
2014-06-19 17:41:51 +00:00
|
|
|
(arch, release, pkg_type) = get_distro_defaults(distro, machine_type)
|
|
|
|
base_url = get_gitbuilder_url('ceph', release, pkg_type, arch,
|
|
|
|
kernel_flavor)
|
2014-06-12 17:30:34 +00:00
|
|
|
url = os.path.join(base_url, 'sha1', hash, 'version')
|
|
|
|
resp = requests.get(url)
|
2014-06-24 15:43:41 +00:00
|
|
|
if resp.ok:
|
|
|
|
return resp.text.strip()
|
2014-06-12 17:30:34 +00:00
|
|
|
|
|
|
|
|
2014-06-19 17:41:51 +00:00
|
|
|
def get_branch_info(project, branch, project_owner='ceph'):
|
2014-06-25 15:32:56 +00:00
|
|
|
"""
|
|
|
|
Use the GitHub API to query a project's branch. Returns:
|
|
|
|
{u'object': {u'sha': <a_sha_string>,
|
|
|
|
u'type': <string>,
|
|
|
|
u'url': <url_to_commit>},
|
|
|
|
u'ref': u'refs/heads/<branch>',
|
|
|
|
u'url': <url_to_branch>}
|
|
|
|
|
|
|
|
We mainly use this to check if a branch exists.
|
|
|
|
"""
|
2014-06-19 17:41:51 +00:00
|
|
|
url_templ = 'https://api.github.com/repos/{project_owner}/{project}/git/refs/heads/{branch}' # noqa
|
|
|
|
url = url_templ.format(project_owner=project_owner, project=project,
|
|
|
|
branch=branch)
|
|
|
|
resp = requests.get(url)
|
|
|
|
if resp.ok:
|
|
|
|
return resp.json()
|
|
|
|
|
|
|
|
|
2014-07-10 01:02:47 +00:00
|
|
|
def schedule_suite(job_config,
|
2014-06-12 16:25:30 +00:00
|
|
|
path,
|
|
|
|
base_yamls,
|
|
|
|
base_args,
|
|
|
|
arch,
|
|
|
|
limit=0,
|
|
|
|
dry_run=True,
|
|
|
|
):
|
2014-06-10 20:50:59 +00:00
|
|
|
"""
|
2014-06-12 16:25:30 +00:00
|
|
|
schedule one suite.
|
2014-06-10 20:50:59 +00:00
|
|
|
returns number of jobs scheduled
|
|
|
|
"""
|
2014-07-10 01:02:47 +00:00
|
|
|
machine_type = job_config.machine_type
|
|
|
|
suite_name = job_config.suite
|
2014-06-10 20:50:59 +00:00
|
|
|
count = 0
|
2014-07-10 01:02:47 +00:00
|
|
|
log.debug('Suite %s in %s' % (suite_name, path))
|
|
|
|
configs = [(combine_path(suite_name, item[0]), item[1]) for item in
|
2014-06-12 16:25:30 +00:00
|
|
|
build_matrix(path)]
|
2014-06-10 20:50:59 +00:00
|
|
|
job_count = len(configs)
|
2014-06-12 16:25:30 +00:00
|
|
|
log.info('Suite %s in %s generated %d jobs' % (
|
2014-07-10 01:02:47 +00:00
|
|
|
suite_name, path, len(configs)))
|
2014-06-10 20:50:59 +00:00
|
|
|
|
2014-06-24 15:13:30 +00:00
|
|
|
for description, fragment_paths in configs:
|
2014-06-12 16:25:30 +00:00
|
|
|
if limit > 0 and count >= limit:
|
2014-06-10 20:50:59 +00:00
|
|
|
log.info(
|
|
|
|
'Stopped after {limit} jobs due to --limit={limit}'.format(
|
|
|
|
limit=limit))
|
|
|
|
break
|
2014-06-24 15:13:30 +00:00
|
|
|
raw_yaml = '\n'.join([file(a, 'r').read() for a in fragment_paths])
|
2014-06-10 20:50:59 +00:00
|
|
|
|
|
|
|
parsed_yaml = yaml.load(raw_yaml)
|
|
|
|
os_type = parsed_yaml.get('os_type')
|
|
|
|
exclude_arch = parsed_yaml.get('exclude_arch')
|
|
|
|
exclude_os_type = parsed_yaml.get('exclude_os_type')
|
|
|
|
|
2014-06-12 17:45:22 +00:00
|
|
|
if exclude_arch and exclude_arch == arch:
|
|
|
|
log.info('Skipping due to excluded_arch: %s facets %s',
|
|
|
|
exclude_arch, description)
|
|
|
|
continue
|
|
|
|
if exclude_os_type and exclude_os_type == os_type:
|
|
|
|
log.info('Skipping due to excluded_os_type: %s facets %s',
|
|
|
|
exclude_os_type, description)
|
|
|
|
continue
|
2014-06-10 20:50:59 +00:00
|
|
|
# We should not run multiple tests (changing distros) unless the
|
|
|
|
# machine is a VPS.
|
|
|
|
# Re-imaging baremetal is not yet supported.
|
2014-06-12 17:45:22 +00:00
|
|
|
if machine_type != 'vps' and os_type and os_type != 'ubuntu':
|
|
|
|
log.info(
|
|
|
|
'Skipping due to non-ubuntu on baremetal facets %s',
|
|
|
|
description)
|
|
|
|
continue
|
2014-06-10 20:50:59 +00:00
|
|
|
|
|
|
|
log.info(
|
|
|
|
'Scheduling %s', description
|
|
|
|
)
|
|
|
|
|
|
|
|
arg = copy.deepcopy(base_args)
|
|
|
|
arg.extend([
|
|
|
|
'--description', description,
|
|
|
|
'--',
|
|
|
|
])
|
|
|
|
arg.extend(base_yamls)
|
2014-06-24 15:13:30 +00:00
|
|
|
arg.extend(fragment_paths)
|
2014-06-10 20:50:59 +00:00
|
|
|
|
|
|
|
if dry_run:
|
2014-06-19 19:54:05 +00:00
|
|
|
# Quote any individual args so that individual commands can be
|
|
|
|
# copied and pasted in order to execute them individually.
|
|
|
|
printable_args = []
|
|
|
|
for item in arg:
|
|
|
|
if ' ' in item:
|
|
|
|
printable_args.append("'%s'" % item)
|
|
|
|
else:
|
|
|
|
printable_args.append(item)
|
|
|
|
log.info('dry-run: %s' % ' '.join(printable_args))
|
2013-09-13 15:39:30 +00:00
|
|
|
else:
|
|
|
|
subprocess.check_call(
|
|
|
|
args=arg,
|
2013-10-08 22:10:01 +00:00
|
|
|
)
|
2014-06-10 20:50:59 +00:00
|
|
|
count += 1
|
|
|
|
return job_count
|
2011-08-26 00:11:33 +00:00
|
|
|
|
2013-08-23 06:29:27 +00:00
|
|
|
|
|
|
|
def combine_path(left, right):
|
|
|
|
"""
|
|
|
|
os.path.join(a, b) doesn't like it when b is None
|
|
|
|
"""
|
|
|
|
if right:
|
|
|
|
return os.path.join(left, right)
|
|
|
|
return left
|
|
|
|
|
2013-10-08 22:10:01 +00:00
|
|
|
|
2013-08-23 06:29:27 +00:00
|
|
|
def build_matrix(path):
|
|
|
|
"""
|
2013-08-28 20:25:22 +00:00
|
|
|
Return a list of items describe by path
|
|
|
|
|
|
|
|
The input is just a path. The output is an array of (description,
|
|
|
|
[file list]) tuples.
|
|
|
|
|
|
|
|
For a normal file we generate a new item for the result list.
|
|
|
|
|
|
|
|
For a directory, we (recursively) generate a new item for each
|
|
|
|
file/dir.
|
|
|
|
|
|
|
|
For a directory with a magic '+' file, we generate a single item
|
|
|
|
that concatenates all files/subdirs.
|
|
|
|
|
|
|
|
For a directory with a magic '%' file, we generate a result set
|
2014-06-30 21:29:48 +00:00
|
|
|
for each item in the directory, and then do a product to generate
|
2013-08-28 20:25:22 +00:00
|
|
|
a result list with all combinations.
|
|
|
|
|
|
|
|
The final description (after recursion) for each item will look
|
|
|
|
like a relative path. If there was a % product, that path
|
|
|
|
component will appear as a file with braces listing the selection
|
|
|
|
of chosen subitems.
|
2013-08-23 06:29:27 +00:00
|
|
|
"""
|
2013-08-28 20:25:22 +00:00
|
|
|
if os.path.isfile(path):
|
2013-08-23 06:29:27 +00:00
|
|
|
if path.endswith('.yaml'):
|
2013-08-28 20:25:22 +00:00
|
|
|
return [(None, [path])]
|
|
|
|
if os.path.isdir(path):
|
2013-08-23 06:29:27 +00:00
|
|
|
files = sorted(os.listdir(path))
|
|
|
|
if '+' in files:
|
|
|
|
# concatenate items
|
|
|
|
files.remove('+')
|
2014-03-07 01:55:00 +00:00
|
|
|
raw = []
|
2013-08-23 06:29:27 +00:00
|
|
|
for fn in files:
|
2014-03-07 01:55:00 +00:00
|
|
|
raw.extend(build_matrix(os.path.join(path, fn)))
|
|
|
|
out = [(
|
|
|
|
'{' + ' '.join(files) + '}',
|
|
|
|
[a[1][0] for a in raw]
|
|
|
|
)]
|
|
|
|
return out
|
2013-08-23 06:29:27 +00:00
|
|
|
elif '%' in files:
|
|
|
|
# convolve items
|
|
|
|
files.remove('%')
|
2013-08-28 20:25:22 +00:00
|
|
|
sublists = []
|
2013-08-23 06:29:27 +00:00
|
|
|
for fn in files:
|
|
|
|
raw = build_matrix(os.path.join(path, fn))
|
2013-10-08 22:10:01 +00:00
|
|
|
sublists.append([(combine_path(fn, item[0]), item[1])
|
|
|
|
for item in raw])
|
2013-08-23 06:29:27 +00:00
|
|
|
out = []
|
2013-09-17 03:22:09 +00:00
|
|
|
if sublists:
|
|
|
|
for sublist in itertools.product(*sublists):
|
|
|
|
name = '{' + ' '.join([item[0] for item in sublist]) + '}'
|
|
|
|
val = []
|
|
|
|
for item in sublist:
|
|
|
|
val.extend(item[1])
|
|
|
|
out.append((name, val))
|
2013-08-23 06:29:27 +00:00
|
|
|
return out
|
|
|
|
else:
|
|
|
|
# list items
|
|
|
|
out = []
|
|
|
|
for fn in files:
|
|
|
|
raw = build_matrix(os.path.join(path, fn))
|
2013-10-08 22:10:01 +00:00
|
|
|
out.extend([(combine_path(fn, item[0]), item[1])
|
|
|
|
for item in raw])
|
2013-08-23 06:29:27 +00:00
|
|
|
return out
|
2013-08-29 04:53:41 +00:00
|
|
|
return []
|
2013-08-23 06:29:27 +00:00
|
|
|
|
2011-06-29 19:54:53 +00:00
|
|
|
|
2014-06-12 17:50:41 +00:00
|
|
|
def get_arch(machine_type):
|
2014-06-25 15:32:56 +00:00
|
|
|
"""
|
|
|
|
Based on a given machine_type, return its architecture by querying the lock
|
|
|
|
server. Sound expensive? It is!
|
|
|
|
|
|
|
|
:returns: A string or None
|
|
|
|
"""
|
2014-06-12 17:50:41 +00:00
|
|
|
locks = lock.list_locks()
|
|
|
|
for machine in locks:
|
|
|
|
if machine['type'] == machine_type:
|
|
|
|
arch = machine['arch']
|
|
|
|
return arch
|
2013-08-22 21:39:56 +00:00
|
|
|
return None
|
2014-06-19 17:41:51 +00:00
|
|
|
|
2014-07-09 17:46:01 +00:00
|
|
|
|
|
|
|
class Placeholder(object):
|
|
|
|
"""
|
|
|
|
A placeholder for use with substitute_placeholders. Simply has a 'name'
|
|
|
|
attribute.
|
|
|
|
"""
|
|
|
|
def __init__(self, name):
|
|
|
|
self.name = name
|
|
|
|
|
|
|
|
|
|
|
|
def substitute_placeholders(input_dict, values_dict):
|
|
|
|
"""
|
|
|
|
Replace any Placeholder instances with values named in values_dict.
|
|
|
|
Searches through nested dicts.
|
|
|
|
|
|
|
|
:param input_dict: A dict which may contain one or more Placeholder
|
|
|
|
instances as values.
|
|
|
|
:param values_dict: A dict, with keys matching the 'name' attributes of all
|
|
|
|
of the Placeholder instances in the input_dict, and
|
|
|
|
values to be substituted.
|
|
|
|
:returns: The modified input_dict
|
|
|
|
"""
|
2014-07-14 21:16:55 +00:00
|
|
|
input_dict = copy.deepcopy(input_dict)
|
|
|
|
|
|
|
|
def _substitute(input_dict, values_dict):
|
|
|
|
for (key, value) in input_dict.iteritems():
|
|
|
|
if isinstance(value, dict):
|
|
|
|
_substitute(value, values_dict)
|
|
|
|
elif isinstance(value, Placeholder):
|
|
|
|
# If there is a Placeholder without a corresponding entry in
|
|
|
|
# values_dict, we will hit a KeyError - we want this.
|
|
|
|
input_dict[key] = values_dict[value.name]
|
|
|
|
return input_dict
|
|
|
|
|
|
|
|
return _substitute(input_dict, values_dict)
|
2014-07-09 17:46:01 +00:00
|
|
|
|
|
|
|
|
|
|
|
# Template for the config that becomes the base for each generated job config
|
|
|
|
dict_templ = {
|
|
|
|
'branch': Placeholder('ceph_branch'),
|
2014-07-10 01:02:47 +00:00
|
|
|
'teuthology_branch': Placeholder('teuthology_branch'),
|
2014-07-09 17:46:01 +00:00
|
|
|
'machine_type': Placeholder('machine_type'),
|
|
|
|
'nuke-on-error': True,
|
2014-07-11 20:40:45 +00:00
|
|
|
'os_type': Placeholder('distro'),
|
2014-07-09 17:46:01 +00:00
|
|
|
'overrides': {
|
|
|
|
'admin_socket': {
|
|
|
|
'branch': Placeholder('ceph_branch'),
|
|
|
|
},
|
|
|
|
'ceph': {
|
|
|
|
'conf': {
|
|
|
|
'mon': {
|
|
|
|
'debug mon': 20,
|
|
|
|
'debug ms': 1,
|
|
|
|
'debug paxos': 20},
|
|
|
|
'osd': {
|
|
|
|
'debug filestore': 20,
|
|
|
|
'debug journal': 20,
|
|
|
|
'debug ms': 1,
|
|
|
|
'debug osd': 20
|
|
|
|
}
|
|
|
|
},
|
|
|
|
'log-whitelist': ['slow request'],
|
|
|
|
'sha1': Placeholder('ceph_hash'),
|
|
|
|
},
|
|
|
|
'ceph-deploy': {
|
|
|
|
'branch': {
|
|
|
|
'dev': Placeholder('ceph_branch'),
|
|
|
|
},
|
|
|
|
'conf': {
|
|
|
|
'client': {
|
|
|
|
'log file': '/var/log/ceph/ceph-$name.$pid.log'
|
|
|
|
},
|
|
|
|
'mon': {
|
|
|
|
'debug mon': 1,
|
|
|
|
'debug ms': 20,
|
|
|
|
'debug paxos': 20,
|
|
|
|
'osd default pool size': 2
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
'install': {
|
|
|
|
'ceph': {
|
|
|
|
'sha1': Placeholder('ceph_hash'),
|
|
|
|
}
|
|
|
|
},
|
|
|
|
's3tests': {
|
|
|
|
'branch': Placeholder('s3_branch'),
|
|
|
|
},
|
|
|
|
'workunit': {
|
|
|
|
'sha1': Placeholder('ceph_hash'),
|
|
|
|
}
|
|
|
|
},
|
2014-07-10 01:02:47 +00:00
|
|
|
'suite': Placeholder('suite'),
|
2014-07-14 20:30:31 +00:00
|
|
|
'suite_branch': Placeholder('suite_branch'),
|
2014-07-09 17:46:01 +00:00
|
|
|
'tasks': [
|
|
|
|
{'chef': None},
|
|
|
|
{'clock.check': None}
|
|
|
|
],
|
|
|
|
}
|