2019-11-08 22:39:00 +00:00
|
|
|
"""
|
|
|
|
Ceph cluster task, deployed via ceph-daemon and ssh orchestrator
|
|
|
|
"""
|
|
|
|
from cStringIO import StringIO
|
|
|
|
|
|
|
|
import argparse
|
|
|
|
import configobj
|
|
|
|
import contextlib
|
|
|
|
import errno
|
|
|
|
import logging
|
|
|
|
import os
|
|
|
|
import json
|
|
|
|
import time
|
|
|
|
import gevent
|
|
|
|
import re
|
|
|
|
import socket
|
|
|
|
import uuid
|
|
|
|
|
|
|
|
from paramiko import SSHException
|
|
|
|
from ceph_manager import CephManager, write_conf
|
|
|
|
from tarfile import ReadError
|
|
|
|
from tasks.cephfs.filesystem import Filesystem
|
|
|
|
from teuthology import misc as teuthology
|
|
|
|
from teuthology import contextutil
|
|
|
|
from teuthology import exceptions
|
|
|
|
from teuthology.orchestra import run
|
|
|
|
import ceph_client as cclient
|
|
|
|
from teuthology.orchestra.daemon import DaemonGroup
|
|
|
|
from tasks.daemonwatchdog import DaemonWatchdog
|
|
|
|
|
|
|
|
# these items we use from ceph.py should probably eventually move elsewhere
|
|
|
|
from tasks.ceph import get_mons
|
|
|
|
|
2019-11-11 15:27:16 +00:00
|
|
|
CEPH_ROLE_TYPES = ['mon', 'mgr', 'osd', 'mds', 'rgw']
|
|
|
|
|
2019-11-08 22:39:00 +00:00
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2019-11-11 20:30:59 +00:00
|
|
|
def shell(ctx, cluster_name, remote, args, **kwargs):
|
2019-11-08 22:39:00 +00:00
|
|
|
testdir = teuthology.get_testdir(ctx)
|
|
|
|
return remote.run(
|
|
|
|
args=[
|
|
|
|
'sudo',
|
|
|
|
'{}/ceph-daemon'.format(testdir),
|
|
|
|
'--image', ctx.image,
|
|
|
|
'shell',
|
2019-11-11 20:30:59 +00:00
|
|
|
'-c', '{}/{}.conf'.format(testdir, cluster_name),
|
|
|
|
'-k', '{}/{}.keyring'.format(testdir, cluster_name),
|
|
|
|
'--fsid', ctx.ceph[cluster_name].fsid,
|
2019-11-08 22:39:00 +00:00
|
|
|
'--',
|
|
|
|
] + args,
|
|
|
|
**kwargs
|
|
|
|
)
|
|
|
|
|
2019-11-11 15:27:16 +00:00
|
|
|
def build_initial_config(ctx, config):
|
2019-11-11 20:30:59 +00:00
|
|
|
cluster_name = config['cluster']
|
|
|
|
|
2019-11-12 22:13:22 +00:00
|
|
|
path = os.path.join(os.path.dirname(__file__), 'ceph2.conf')
|
|
|
|
conf = configobj.ConfigObj(path, file_error=True)
|
2019-11-11 15:27:16 +00:00
|
|
|
|
|
|
|
conf.setdefault('global', {})
|
2019-11-11 20:30:59 +00:00
|
|
|
conf['global']['fsid'] = ctx.ceph[cluster_name].fsid
|
2019-11-11 15:27:16 +00:00
|
|
|
|
|
|
|
# overrides
|
|
|
|
for section, keys in config['conf'].items():
|
|
|
|
for key, value in keys.items():
|
2019-11-12 22:13:22 +00:00
|
|
|
log.info(" override: [%s] %s = %s" % (section, key, value))
|
2019-11-11 15:27:16 +00:00
|
|
|
if section not in conf:
|
|
|
|
conf[section] = {}
|
|
|
|
conf[section][key] = value
|
|
|
|
|
|
|
|
return conf
|
|
|
|
|
|
|
|
@contextlib.contextmanager
|
|
|
|
def normalize_hostnames(ctx):
|
|
|
|
"""
|
|
|
|
Ensure we have short hostnames throughout, for consistency between
|
|
|
|
remote.shortname and socket.gethostname() in ceph-daemon.
|
|
|
|
"""
|
|
|
|
log.info('Normalizing hostnames...')
|
|
|
|
ctx.cluster.run(args=[
|
|
|
|
'sudo',
|
|
|
|
'hostname',
|
|
|
|
run.Raw('$(hostname -s)'),
|
|
|
|
])
|
|
|
|
|
|
|
|
try:
|
|
|
|
yield
|
|
|
|
finally:
|
|
|
|
pass
|
|
|
|
|
2019-11-08 22:39:00 +00:00
|
|
|
@contextlib.contextmanager
|
|
|
|
def download_ceph_daemon(ctx, config):
|
2019-11-11 20:30:59 +00:00
|
|
|
cluster_name = config['cluster']
|
2019-11-08 22:39:00 +00:00
|
|
|
testdir = teuthology.get_testdir(ctx)
|
|
|
|
branch = config.get('ceph-daemon-branch', 'master')
|
|
|
|
|
2019-11-11 20:30:59 +00:00
|
|
|
log.info('Downloading ceph-daemon...')
|
2019-11-08 22:39:00 +00:00
|
|
|
ctx.cluster.run(
|
|
|
|
args=[
|
|
|
|
'curl', '--silent',
|
|
|
|
'https://raw.githubusercontent.com/ceph/ceph/%s/src/ceph-daemon/ceph-daemon' % branch,
|
|
|
|
run.Raw('>'),
|
|
|
|
'{tdir}/ceph-daemon'.format(tdir=testdir),
|
|
|
|
run.Raw('&&'),
|
|
|
|
'test', '-s',
|
|
|
|
'{tdir}/ceph-daemon'.format(tdir=testdir),
|
|
|
|
run.Raw('&&'),
|
|
|
|
'chmod', '+x',
|
|
|
|
'{tdir}/ceph-daemon'.format(tdir=testdir),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
|
|
|
|
try:
|
|
|
|
yield
|
|
|
|
finally:
|
2019-11-11 15:27:16 +00:00
|
|
|
log.info('Removing cluster...')
|
|
|
|
ctx.cluster.run(args=[
|
|
|
|
'sudo',
|
|
|
|
'{}/ceph-daemon'.format(testdir),
|
|
|
|
'rm-cluster',
|
2019-11-11 20:30:59 +00:00
|
|
|
'--fsid', ctx.ceph[cluster_name].fsid,
|
2019-11-11 15:27:16 +00:00
|
|
|
'--force',
|
|
|
|
])
|
|
|
|
|
2019-11-08 22:39:00 +00:00
|
|
|
log.info('Removing ceph-daemon ...')
|
|
|
|
ctx.cluster.run(
|
|
|
|
args=[
|
|
|
|
'rm',
|
|
|
|
'-rf',
|
|
|
|
'{tdir}/ceph-daemon'.format(tdir=testdir),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
|
|
|
|
@contextlib.contextmanager
|
2019-11-11 20:30:59 +00:00
|
|
|
def ceph_log(ctx, config):
|
|
|
|
cluster_name = config['cluster']
|
|
|
|
fsid = ctx.ceph[cluster_name].fsid
|
|
|
|
|
2019-11-08 22:39:00 +00:00
|
|
|
try:
|
|
|
|
yield
|
|
|
|
|
|
|
|
finally:
|
|
|
|
if ctx.archive is not None and \
|
|
|
|
not (ctx.config.get('archive-on-error') and ctx.summary['success']):
|
|
|
|
# and logs
|
|
|
|
log.info('Compressing logs...')
|
|
|
|
run.wait(
|
|
|
|
ctx.cluster.run(
|
|
|
|
args=[
|
|
|
|
'sudo',
|
|
|
|
'find',
|
|
|
|
'/var/log/ceph/' + fsid,
|
|
|
|
'-name',
|
|
|
|
'*.log',
|
|
|
|
'-print0',
|
|
|
|
run.Raw('|'),
|
|
|
|
'sudo',
|
|
|
|
'xargs',
|
|
|
|
'-0',
|
|
|
|
'--no-run-if-empty',
|
|
|
|
'--',
|
|
|
|
'gzip',
|
|
|
|
'--',
|
|
|
|
],
|
|
|
|
wait=False,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
log.info('Archiving logs...')
|
|
|
|
path = os.path.join(ctx.archive, 'remote')
|
|
|
|
try:
|
|
|
|
os.makedirs(path)
|
|
|
|
except OSError as e:
|
|
|
|
pass
|
|
|
|
for remote in ctx.cluster.remotes.keys():
|
2019-11-11 15:27:16 +00:00
|
|
|
sub = os.path.join(path, remote.name)
|
2019-11-08 22:39:00 +00:00
|
|
|
try:
|
|
|
|
os.makedirs(sub)
|
|
|
|
except OSError as e:
|
|
|
|
pass
|
|
|
|
teuthology.pull_directory(remote, '/var/log/ceph/' + fsid,
|
|
|
|
os.path.join(sub, 'log'))
|
|
|
|
|
|
|
|
@contextlib.contextmanager
|
2019-11-11 20:30:59 +00:00
|
|
|
def ceph_crash(ctx, config):
|
2019-11-08 22:39:00 +00:00
|
|
|
"""
|
|
|
|
Gather crash dumps from /var/lib/ceph/$fsid/crash
|
|
|
|
"""
|
2019-11-11 20:30:59 +00:00
|
|
|
cluster_name = config['cluster']
|
|
|
|
fsid = ctx.ceph[cluster_name].fsid
|
|
|
|
|
2019-11-08 22:39:00 +00:00
|
|
|
try:
|
|
|
|
yield
|
|
|
|
|
|
|
|
finally:
|
|
|
|
if ctx.archive is not None:
|
|
|
|
log.info('Archiving crash dumps...')
|
|
|
|
path = os.path.join(ctx.archive, 'remote')
|
|
|
|
try:
|
|
|
|
os.makedirs(path)
|
|
|
|
except OSError as e:
|
|
|
|
pass
|
|
|
|
for remote in ctx.cluster.remotes.keys():
|
2019-11-11 15:27:16 +00:00
|
|
|
sub = os.path.join(path, remote.name)
|
2019-11-08 22:39:00 +00:00
|
|
|
try:
|
|
|
|
os.makedirs(sub)
|
|
|
|
except OSError as e:
|
|
|
|
pass
|
|
|
|
try:
|
|
|
|
teuthology.pull_directory(remote,
|
|
|
|
'/var/lib/ceph/%s/crash' % fsid,
|
|
|
|
os.path.join(sub, 'crash'))
|
|
|
|
except ReadError as e:
|
|
|
|
pass
|
|
|
|
|
|
|
|
@contextlib.contextmanager
|
2019-11-11 20:30:59 +00:00
|
|
|
def ceph_bootstrap(ctx, config):
|
|
|
|
cluster_name = config['cluster']
|
2019-11-08 22:39:00 +00:00
|
|
|
testdir = teuthology.get_testdir(ctx)
|
2019-11-11 20:30:59 +00:00
|
|
|
fsid = ctx.ceph[cluster_name].fsid
|
2019-11-08 22:39:00 +00:00
|
|
|
|
2019-11-11 20:30:59 +00:00
|
|
|
mons = ctx.ceph[cluster_name].mons
|
|
|
|
first_mon_role = sorted(mons.keys())[0]
|
|
|
|
_, _, first_mon = teuthology.split_role(first_mon_role)
|
2019-11-11 23:02:19 +00:00
|
|
|
(bootstrap_remote,) = ctx.cluster.only(first_mon_role).remotes.keys()
|
|
|
|
log.info('First mon is mon.%s on %s' % (first_mon,
|
|
|
|
bootstrap_remote.shortname))
|
|
|
|
ctx.ceph[cluster_name].bootstrap_remote = bootstrap_remote
|
2019-11-11 20:30:59 +00:00
|
|
|
ctx.ceph[cluster_name].first_mon = first_mon
|
2019-11-08 22:39:00 +00:00
|
|
|
|
2019-11-11 23:02:19 +00:00
|
|
|
others = ctx.cluster.remotes[bootstrap_remote]
|
2019-11-08 22:39:00 +00:00
|
|
|
log.info('others %s' % others)
|
2019-11-11 20:30:59 +00:00
|
|
|
mgrs = sorted([r for r in others
|
|
|
|
if teuthology.is_type('mgr', cluster_name)(r)])
|
2019-11-08 22:39:00 +00:00
|
|
|
if not mgrs:
|
|
|
|
raise RuntimeError('no mgrs on the same host as first mon %s' % first_mon)
|
2019-11-11 20:30:59 +00:00
|
|
|
_, _, first_mgr = teuthology.split_role(mgrs[0])
|
2019-11-08 22:39:00 +00:00
|
|
|
log.info('First mgr is %s' % (first_mgr))
|
2019-11-11 20:30:59 +00:00
|
|
|
ctx.ceph[cluster_name].first_mgr = first_mgr
|
2019-11-08 22:39:00 +00:00
|
|
|
|
|
|
|
try:
|
2019-11-11 15:27:16 +00:00
|
|
|
# write seed config
|
|
|
|
log.info('Writing seed config...')
|
|
|
|
conf_fp = StringIO()
|
|
|
|
seed_config = build_initial_config(ctx, config)
|
|
|
|
seed_config.write(conf_fp)
|
|
|
|
teuthology.write_file(
|
2019-11-11 23:02:19 +00:00
|
|
|
remote=bootstrap_remote,
|
2019-11-11 20:30:59 +00:00
|
|
|
path='{}/seed.{}.conf'.format(testdir, cluster_name),
|
2019-11-11 15:27:16 +00:00
|
|
|
data=conf_fp.getvalue())
|
2019-11-12 22:13:22 +00:00
|
|
|
log.debug('Final config:\n' + conf_fp.getvalue())
|
2019-11-11 15:27:16 +00:00
|
|
|
|
|
|
|
# bootstrap
|
|
|
|
log.info('Bootstrapping...')
|
2019-11-08 22:39:00 +00:00
|
|
|
cmd = [
|
|
|
|
'sudo',
|
|
|
|
'{}/ceph-daemon'.format(testdir),
|
|
|
|
'--image', ctx.image,
|
|
|
|
'bootstrap',
|
|
|
|
'--fsid', fsid,
|
2019-11-11 20:30:59 +00:00
|
|
|
'--mon-id', first_mon,
|
|
|
|
'--mgr-id', first_mgr,
|
|
|
|
'--config', '{}/seed.{}.conf'.format(testdir, cluster_name),
|
|
|
|
'--output-config', '{}/{}.conf'.format(testdir, cluster_name),
|
|
|
|
'--output-keyring', '{}/{}.keyring'.format(testdir, cluster_name),
|
|
|
|
'--output-pub-ssh-key', '{}/{}.pub'.format(testdir, cluster_name),
|
2019-11-08 22:39:00 +00:00
|
|
|
]
|
2019-11-11 20:30:59 +00:00
|
|
|
if mons[first_mon_role].startswith('['):
|
|
|
|
cmd += ['--mon-addrv', mons[first_mon_role]]
|
2019-11-08 22:39:00 +00:00
|
|
|
else:
|
2019-11-11 20:30:59 +00:00
|
|
|
cmd += ['--mon-ip', mons[first_mon_role]]
|
2019-11-11 15:27:16 +00:00
|
|
|
if config.get('skip_dashboard'):
|
|
|
|
cmd += ['--skip-dashboard']
|
|
|
|
# bootstrap makes the keyring root 0600, so +r it for our purposes
|
2019-11-08 22:39:00 +00:00
|
|
|
cmd += [
|
|
|
|
run.Raw('&&'),
|
2019-11-11 20:30:59 +00:00
|
|
|
'sudo', 'chmod', '+r', '{}/{}.keyring'.format(testdir, cluster_name),
|
2019-11-08 22:39:00 +00:00
|
|
|
]
|
2019-11-11 23:02:19 +00:00
|
|
|
bootstrap_remote.run(args=cmd)
|
|
|
|
|
|
|
|
# register initial daemons
|
|
|
|
ctx.daemons.register_daemon(
|
|
|
|
bootstrap_remote, 'mon', first_mon,
|
|
|
|
cluster=cluster_name,
|
|
|
|
fsid=fsid,
|
|
|
|
logger=log.getChild('mon.' + first_mon),
|
|
|
|
wait=False,
|
|
|
|
started=True,
|
|
|
|
)
|
|
|
|
ctx.daemons.register_daemon(
|
|
|
|
bootstrap_remote, 'mgr', first_mgr,
|
|
|
|
cluster=cluster_name,
|
|
|
|
fsid=fsid,
|
|
|
|
logger=log.getChild('mgr.' + first_mgr),
|
|
|
|
wait=False,
|
|
|
|
started=True,
|
|
|
|
)
|
2019-11-08 22:39:00 +00:00
|
|
|
|
|
|
|
# fetch keys and configs
|
|
|
|
log.info('Fetching config...')
|
2019-11-11 20:30:59 +00:00
|
|
|
ctx.ceph[cluster_name].config_file = teuthology.get_file(
|
2019-11-11 23:02:19 +00:00
|
|
|
remote=bootstrap_remote,
|
2019-11-11 20:30:59 +00:00
|
|
|
path='{}/{}.conf'.format(testdir, cluster_name))
|
2019-11-11 15:27:16 +00:00
|
|
|
log.info('Fetching client.admin keyring...')
|
2019-11-11 20:30:59 +00:00
|
|
|
ctx.ceph[cluster_name].admin_keyring = teuthology.get_file(
|
2019-11-11 23:02:19 +00:00
|
|
|
remote=bootstrap_remote,
|
2019-11-11 20:30:59 +00:00
|
|
|
path='{}/{}.keyring'.format(testdir, cluster_name))
|
2019-11-08 22:39:00 +00:00
|
|
|
log.info('Fetching mon keyring...')
|
2019-11-11 20:30:59 +00:00
|
|
|
ctx.ceph[cluster_name].mon_keyring = teuthology.get_file(
|
2019-11-11 23:02:19 +00:00
|
|
|
remote=bootstrap_remote,
|
2019-11-11 20:30:59 +00:00
|
|
|
path='/var/lib/ceph/%s/mon.%s/keyring' % (fsid, first_mon),
|
2019-11-08 22:39:00 +00:00
|
|
|
sudo=True)
|
|
|
|
|
|
|
|
# fetch ssh key, distribute to additional nodes
|
|
|
|
log.info('Fetching pub ssh key...')
|
|
|
|
ssh_pub_key = teuthology.get_file(
|
2019-11-11 23:02:19 +00:00
|
|
|
remote=bootstrap_remote,
|
2019-11-11 20:30:59 +00:00
|
|
|
path='{}/{}.pub'.format(testdir, cluster_name)
|
2019-11-08 22:39:00 +00:00
|
|
|
).strip()
|
|
|
|
|
|
|
|
log.info('Installing pub ssh key for root users...')
|
|
|
|
ctx.cluster.run(args=[
|
|
|
|
'sudo', 'install', '-d', '-m', '0700', '/root/.ssh',
|
|
|
|
run.Raw('&&'),
|
|
|
|
'echo', ssh_pub_key,
|
|
|
|
run.Raw('|'),
|
|
|
|
'sudo', 'tee', '-a', '/root/.ssh/authorized_keys',
|
|
|
|
run.Raw('&&'),
|
|
|
|
'sudo', 'chmod', '0600', '/root/.ssh/authorized_keys',
|
|
|
|
])
|
|
|
|
|
|
|
|
# add other hosts
|
|
|
|
for remote in ctx.cluster.remotes.keys():
|
2019-11-11 23:02:19 +00:00
|
|
|
if remote == bootstrap_remote:
|
2019-11-08 22:39:00 +00:00
|
|
|
continue
|
|
|
|
log.info('Writing conf and keyring to %s' % remote.shortname)
|
|
|
|
teuthology.write_file(
|
|
|
|
remote=remote,
|
2019-11-11 20:30:59 +00:00
|
|
|
path='{}/{}.conf'.format(testdir, cluster_name),
|
|
|
|
data=ctx.ceph[cluster_name].config_file)
|
2019-11-08 22:39:00 +00:00
|
|
|
teuthology.write_file(
|
|
|
|
remote=remote,
|
2019-11-11 20:30:59 +00:00
|
|
|
path='{}/{}.keyring'.format(testdir, cluster_name),
|
|
|
|
data=ctx.ceph[cluster_name].admin_keyring)
|
2019-11-08 22:39:00 +00:00
|
|
|
|
|
|
|
log.info('Adding host %s to orchestrator...' % remote.shortname)
|
2019-11-11 20:30:59 +00:00
|
|
|
shell(ctx, cluster_name, remote, [
|
2019-11-08 22:39:00 +00:00
|
|
|
'ceph', 'orchestrator', 'host', 'add',
|
|
|
|
remote.shortname
|
|
|
|
])
|
|
|
|
|
|
|
|
yield
|
|
|
|
|
|
|
|
finally:
|
|
|
|
log.info('Cleaning up testdir ceph.* files...')
|
|
|
|
ctx.cluster.run(args=[
|
|
|
|
'rm', '-f',
|
2019-11-11 20:30:59 +00:00
|
|
|
'{}/seed.{}.conf'.format(testdir, cluster_name),
|
|
|
|
'{}/{}.pub'.format(testdir, cluster_name),
|
|
|
|
'{}/{}.conf'.format(testdir, cluster_name),
|
|
|
|
'{}/{}.keyring'.format(testdir, cluster_name),
|
2019-11-08 22:39:00 +00:00
|
|
|
])
|
2019-11-11 15:27:16 +00:00
|
|
|
|
|
|
|
log.info('Stopping all daemons...')
|
2019-11-11 20:30:59 +00:00
|
|
|
|
|
|
|
# this doesn't block until they are all stopped...
|
|
|
|
#ctx.cluster.run(args=['sudo', 'systemctl', 'stop', 'ceph.target'])
|
|
|
|
|
|
|
|
# so, stop them individually
|
2019-11-11 23:02:19 +00:00
|
|
|
for role in ctx.daemons.resolve_role_list(None, CEPH_ROLE_TYPES):
|
|
|
|
cluster, type_, id_ = teuthology.split_role(role)
|
|
|
|
ctx.daemons.get_daemon(type_, id_, cluster).stop()
|
2019-11-08 22:39:00 +00:00
|
|
|
|
|
|
|
@contextlib.contextmanager
|
|
|
|
def ceph_mons(ctx, config):
|
|
|
|
"""
|
|
|
|
Deploy any additional mons
|
|
|
|
"""
|
2019-11-11 20:30:59 +00:00
|
|
|
cluster_name = config['cluster']
|
2019-11-11 23:02:19 +00:00
|
|
|
fsid = ctx.ceph[cluster_name].fsid
|
2019-11-08 22:39:00 +00:00
|
|
|
testdir = teuthology.get_testdir(ctx)
|
|
|
|
num_mons = 1
|
|
|
|
|
|
|
|
try:
|
|
|
|
for remote, roles in ctx.cluster.remotes.items():
|
2019-11-11 20:30:59 +00:00
|
|
|
for mon in [r for r in roles
|
|
|
|
if teuthology.is_type('mon', cluster_name)(r)]:
|
|
|
|
c_, _, id_ = teuthology.split_role(mon)
|
|
|
|
if c_ == cluster_name and id_ == ctx.ceph[cluster_name].first_mon:
|
2019-11-08 22:39:00 +00:00
|
|
|
continue
|
2019-11-11 23:02:19 +00:00
|
|
|
log.info('Adding %s on %s' % (mon, remote.shortname))
|
2019-11-08 22:39:00 +00:00
|
|
|
num_mons += 1
|
2019-11-11 20:30:59 +00:00
|
|
|
shell(ctx, cluster_name, remote, [
|
2019-11-08 22:39:00 +00:00
|
|
|
'ceph', 'orchestrator', 'mon', 'update',
|
|
|
|
str(num_mons),
|
2019-11-11 23:02:19 +00:00
|
|
|
remote.shortname + ':' + ctx.ceph[cluster_name].mons[mon] + '=' + id_,
|
2019-11-08 22:39:00 +00:00
|
|
|
])
|
2019-11-11 23:02:19 +00:00
|
|
|
ctx.daemons.register_daemon(
|
|
|
|
remote, 'mon', id_,
|
|
|
|
cluster=cluster_name,
|
|
|
|
fsid=fsid,
|
|
|
|
logger=log.getChild(mon),
|
|
|
|
wait=False,
|
|
|
|
started=True,
|
|
|
|
)
|
2019-11-08 22:39:00 +00:00
|
|
|
|
|
|
|
while True:
|
|
|
|
log.info('Waiting for %d mons in monmap...' % (num_mons))
|
|
|
|
r = shell(
|
|
|
|
ctx=ctx,
|
2019-11-11 20:30:59 +00:00
|
|
|
cluster_name=cluster_name,
|
2019-11-11 15:27:16 +00:00
|
|
|
remote=remote,
|
2019-11-08 22:39:00 +00:00
|
|
|
args=[
|
|
|
|
'ceph', 'mon', 'dump', '-f', 'json',
|
|
|
|
],
|
|
|
|
stdout=StringIO(),
|
|
|
|
)
|
|
|
|
j = json.loads(r.stdout.getvalue())
|
|
|
|
if len(j['mons']) == num_mons:
|
|
|
|
break
|
|
|
|
time.sleep(1)
|
|
|
|
|
2019-11-11 15:27:16 +00:00
|
|
|
## FIXME: refresh ceph.conf files for all mons + first mgr ##
|
|
|
|
|
2019-11-08 22:39:00 +00:00
|
|
|
yield
|
|
|
|
|
|
|
|
finally:
|
|
|
|
pass
|
|
|
|
|
|
|
|
@contextlib.contextmanager
|
|
|
|
def ceph_mgrs(ctx, config):
|
|
|
|
"""
|
|
|
|
Deploy any additional mgrs
|
|
|
|
"""
|
2019-11-11 20:30:59 +00:00
|
|
|
cluster_name = config['cluster']
|
2019-11-11 23:02:19 +00:00
|
|
|
fsid = ctx.ceph[cluster_name].fsid
|
2019-11-08 22:39:00 +00:00
|
|
|
testdir = teuthology.get_testdir(ctx)
|
|
|
|
|
|
|
|
try:
|
|
|
|
nodes = []
|
|
|
|
for remote, roles in ctx.cluster.remotes.items():
|
2019-11-11 20:30:59 +00:00
|
|
|
for mgr in [r for r in roles
|
|
|
|
if teuthology.is_type('mgr', cluster_name)(r)]:
|
|
|
|
c_, _, id_ = teuthology.split_role(mgr)
|
|
|
|
if c_ == cluster_name and id_ == ctx.ceph[cluster_name].first_mgr:
|
2019-11-11 15:27:16 +00:00
|
|
|
continue
|
2019-11-11 23:02:19 +00:00
|
|
|
log.info('Adding %s on %s' % (mgr, remote.shortname))
|
|
|
|
nodes.append(remote.shortname + '=' + id_)
|
|
|
|
|
|
|
|
ctx.daemons.register_daemon(
|
|
|
|
remote, 'mgr', id_,
|
|
|
|
cluster=cluster_name,
|
|
|
|
fsid=fsid,
|
|
|
|
logger=log.getChild(mgr),
|
|
|
|
wait=False,
|
|
|
|
started=True,
|
|
|
|
)
|
|
|
|
|
2019-11-11 20:30:59 +00:00
|
|
|
shell(ctx, cluster_name, remote, [
|
2019-11-08 22:39:00 +00:00
|
|
|
'ceph', 'orchestrator', 'mgr', 'update',
|
|
|
|
str(len(nodes) + 1)] + nodes
|
|
|
|
)
|
|
|
|
|
|
|
|
yield
|
|
|
|
|
|
|
|
finally:
|
|
|
|
pass
|
|
|
|
|
|
|
|
@contextlib.contextmanager
|
2019-11-11 15:27:16 +00:00
|
|
|
def ceph_osds(ctx, config):
|
|
|
|
"""
|
|
|
|
Deploy OSDs
|
|
|
|
"""
|
2019-11-11 20:30:59 +00:00
|
|
|
cluster_name = config['cluster']
|
2019-11-11 23:02:19 +00:00
|
|
|
fsid = ctx.ceph[cluster_name].fsid
|
2019-11-11 15:27:16 +00:00
|
|
|
try:
|
|
|
|
log.info('Deploying OSDs...')
|
2019-11-11 23:02:19 +00:00
|
|
|
|
2019-11-12 01:22:57 +00:00
|
|
|
# provision OSDs in numeric order
|
2019-11-11 23:02:19 +00:00
|
|
|
id_to_remote = {}
|
|
|
|
devs_by_remote = {}
|
2019-11-11 15:27:16 +00:00
|
|
|
for remote, roles in ctx.cluster.remotes.items():
|
2019-11-11 23:02:19 +00:00
|
|
|
devs_by_remote[remote] = teuthology.get_scratch_devices(remote)
|
|
|
|
for osd in [r for r in roles
|
|
|
|
if teuthology.is_type('osd', cluster_name)(r)]:
|
|
|
|
_, _, id_ = teuthology.split_role(osd)
|
2019-11-12 01:22:57 +00:00
|
|
|
id_to_remote[int(id_)] = (osd, remote)
|
2019-11-11 23:02:19 +00:00
|
|
|
|
|
|
|
cur = 0
|
2019-11-12 01:22:57 +00:00
|
|
|
for osd_id in sorted(id_to_remote.keys()):
|
|
|
|
osd, remote = id_to_remote[osd_id]
|
2019-11-11 23:02:19 +00:00
|
|
|
_, _, id_ = teuthology.split_role(osd)
|
|
|
|
assert int(id_) == cur
|
|
|
|
devs = devs_by_remote[remote]
|
|
|
|
assert devs ## FIXME ##
|
|
|
|
dev = devs.pop()
|
|
|
|
log.info('Deploying %s on %s with %s...' % (
|
|
|
|
osd, remote.shortname, dev))
|
|
|
|
shell(ctx, cluster_name, remote, [
|
|
|
|
'ceph-volume', 'lvm', 'zap', dev])
|
|
|
|
shell(ctx, cluster_name, remote, [
|
|
|
|
'ceph', 'orchestrator', 'osd', 'create',
|
|
|
|
remote.shortname + ':' + dev
|
|
|
|
])
|
|
|
|
ctx.daemons.register_daemon(
|
|
|
|
remote, 'osd', id_,
|
|
|
|
cluster=cluster_name,
|
|
|
|
fsid=fsid,
|
|
|
|
logger=log.getChild(osd),
|
|
|
|
wait=False,
|
|
|
|
started=True,
|
|
|
|
)
|
|
|
|
cur += 1
|
2019-11-11 15:27:16 +00:00
|
|
|
|
|
|
|
yield
|
|
|
|
finally:
|
|
|
|
pass
|
|
|
|
|
2019-11-11 23:14:17 +00:00
|
|
|
@contextlib.contextmanager
|
|
|
|
def ceph_mdss(ctx, config):
|
|
|
|
"""
|
|
|
|
Deploy MDSss
|
|
|
|
"""
|
|
|
|
cluster_name = config['cluster']
|
|
|
|
fsid = ctx.ceph[cluster_name].fsid
|
|
|
|
testdir = teuthology.get_testdir(ctx)
|
|
|
|
|
|
|
|
nodes = []
|
|
|
|
for remote, roles in ctx.cluster.remotes.items():
|
|
|
|
for role in [r for r in roles
|
|
|
|
if teuthology.is_type('mds', cluster_name)(r)]:
|
|
|
|
c_, _, id_ = teuthology.split_role(role)
|
|
|
|
log.info('Adding %s on %s' % (role, remote.shortname))
|
2019-11-12 13:31:52 +00:00
|
|
|
nodes.append(remote.shortname + '=' + id_)
|
2019-11-11 23:14:17 +00:00
|
|
|
ctx.daemons.register_daemon(
|
|
|
|
remote, 'mds', id_,
|
|
|
|
cluster=cluster_name,
|
|
|
|
fsid=fsid,
|
|
|
|
logger=log.getChild(role),
|
|
|
|
wait=False,
|
|
|
|
started=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
shell(ctx, cluster_name, remote, [
|
|
|
|
'ceph', 'orchestrator', 'mds', 'update',
|
|
|
|
'all',
|
|
|
|
str(len(nodes))] + nodes
|
|
|
|
)
|
|
|
|
|
|
|
|
yield
|
|
|
|
|
2019-11-11 15:27:16 +00:00
|
|
|
@contextlib.contextmanager
|
|
|
|
def ceph_initial():
|
2019-11-08 22:39:00 +00:00
|
|
|
try:
|
|
|
|
yield
|
|
|
|
finally:
|
2019-11-11 15:27:16 +00:00
|
|
|
log.info('Teardown complete')
|
|
|
|
|
|
|
|
## public methods
|
|
|
|
@contextlib.contextmanager
|
|
|
|
def stop(ctx, config):
|
|
|
|
"""
|
|
|
|
Stop ceph daemons
|
|
|
|
|
|
|
|
For example::
|
|
|
|
tasks:
|
|
|
|
- ceph.stop: [mds.*]
|
|
|
|
|
|
|
|
tasks:
|
|
|
|
- ceph.stop: [osd.0, osd.2]
|
|
|
|
|
|
|
|
tasks:
|
|
|
|
- ceph.stop:
|
|
|
|
daemons: [osd.0, osd.2]
|
|
|
|
|
|
|
|
"""
|
|
|
|
if config is None:
|
|
|
|
config = {}
|
|
|
|
elif isinstance(config, list):
|
|
|
|
config = {'daemons': config}
|
|
|
|
|
|
|
|
daemons = ctx.daemons.resolve_role_list(
|
|
|
|
config.get('daemons', None), CEPH_ROLE_TYPES, True)
|
|
|
|
clusters = set()
|
|
|
|
|
|
|
|
for role in daemons:
|
|
|
|
cluster, type_, id_ = teuthology.split_role(role)
|
|
|
|
ctx.daemons.get_daemon(type_, id_, cluster).stop()
|
|
|
|
clusters.add(cluster)
|
|
|
|
|
|
|
|
# for cluster in clusters:
|
|
|
|
# ctx.ceph[cluster].watchdog.stop()
|
|
|
|
# ctx.ceph[cluster].watchdog.join()
|
|
|
|
|
|
|
|
yield
|
|
|
|
|
2019-11-11 20:30:59 +00:00
|
|
|
@contextlib.contextmanager
|
|
|
|
def distribute_config_and_admin_keyring(ctx, config):
|
|
|
|
"""
|
|
|
|
Distribute a sufficient config and keyring for clients
|
|
|
|
"""
|
|
|
|
cluster_name = config['cluster']
|
|
|
|
log.info('Distributing config and client.admin keyring...')
|
|
|
|
for remote, roles in ctx.cluster.remotes.items():
|
|
|
|
remote.run(args=['sudo', 'mkdir', '-p', '/etc/ceph'])
|
|
|
|
teuthology.sudo_write_file(
|
|
|
|
remote=remote,
|
|
|
|
path='/etc/ceph/{}.conf'.format(cluster_name),
|
|
|
|
data=ctx.ceph[cluster_name].config_file)
|
|
|
|
teuthology.sudo_write_file(
|
|
|
|
remote=remote,
|
|
|
|
path='/etc/ceph/{}.client.admin.keyring'.format(cluster_name),
|
|
|
|
data=ctx.ceph[cluster_name].admin_keyring)
|
|
|
|
try:
|
|
|
|
yield
|
|
|
|
finally:
|
|
|
|
ctx.cluster.run(args=[
|
|
|
|
'sudo', 'rm', '-f',
|
|
|
|
'/etc/ceph/{}.conf'.format(cluster_name),
|
|
|
|
'/etc/ceph/{}.client.admin.keyring'.format(cluster_name),
|
|
|
|
])
|
2019-11-08 22:39:00 +00:00
|
|
|
|
|
|
|
@contextlib.contextmanager
|
|
|
|
def task(ctx, config):
|
|
|
|
if config is None:
|
|
|
|
config = {}
|
|
|
|
|
|
|
|
assert isinstance(config, dict), \
|
|
|
|
"task only supports a dictionary for configuration"
|
|
|
|
|
|
|
|
overrides = ctx.config.get('overrides', {})
|
|
|
|
teuthology.deep_merge(config, overrides.get('ceph', {}))
|
2019-11-11 15:27:16 +00:00
|
|
|
log.info('Config: ' + str(config))
|
2019-11-08 22:39:00 +00:00
|
|
|
|
2019-11-11 15:27:16 +00:00
|
|
|
testdir = teuthology.get_testdir(ctx)
|
2019-11-08 22:39:00 +00:00
|
|
|
|
2019-11-11 20:30:59 +00:00
|
|
|
# set up cluster context
|
2019-11-08 22:39:00 +00:00
|
|
|
first_ceph_cluster = False
|
|
|
|
if not hasattr(ctx, 'daemons'):
|
|
|
|
first_ceph_cluster = True
|
2019-11-11 20:30:59 +00:00
|
|
|
ctx.daemons = DaemonGroup(use_ceph_daemon=True)
|
2019-11-08 22:39:00 +00:00
|
|
|
if not hasattr(ctx, 'ceph'):
|
|
|
|
ctx.ceph = {}
|
2019-11-12 22:15:32 +00:00
|
|
|
ctx.managers = {}
|
2019-11-08 22:39:00 +00:00
|
|
|
if 'cluster' not in config:
|
|
|
|
config['cluster'] = 'ceph'
|
|
|
|
cluster_name = config['cluster']
|
|
|
|
ctx.ceph[cluster_name] = argparse.Namespace()
|
|
|
|
|
|
|
|
#validate_config(ctx, config)
|
|
|
|
|
|
|
|
# image
|
|
|
|
branch = config.get('branch', 'master')
|
|
|
|
### FIXME ###
|
|
|
|
if branch in ['master', 'nautilus']:
|
|
|
|
ctx.image = 'ceph/daemon-base:latest-%s-devel' % branch
|
|
|
|
else:
|
2019-11-11 15:27:16 +00:00
|
|
|
# ctx.image = 'ceph-ci/ceph:%s' % branch
|
|
|
|
ctx.image = 'cephci/daemon-base:%s' % branch
|
2019-11-08 22:39:00 +00:00
|
|
|
log.info('Cluster image is %s' % ctx.image)
|
|
|
|
|
2019-11-11 15:27:16 +00:00
|
|
|
# uuid
|
2019-11-08 22:39:00 +00:00
|
|
|
fsid = str(uuid.uuid1())
|
|
|
|
log.info('Cluster fsid is %s' % fsid)
|
|
|
|
ctx.ceph[cluster_name].fsid = fsid
|
|
|
|
|
|
|
|
# mon ips
|
|
|
|
log.info('Choosing monitor IPs and ports...')
|
|
|
|
remotes_and_roles = ctx.cluster.remotes.items()
|
|
|
|
roles = [role_list for (remote, role_list) in remotes_and_roles]
|
|
|
|
ips = [host for (host, port) in
|
|
|
|
(remote.ssh.get_transport().getpeername() for (remote, role_list) in remotes_and_roles)]
|
2019-11-11 20:30:59 +00:00
|
|
|
ctx.ceph[cluster_name].mons = get_mons(
|
2019-11-08 22:39:00 +00:00
|
|
|
roles, ips, cluster_name,
|
|
|
|
mon_bind_msgr2=config.get('mon_bind_msgr2', True),
|
|
|
|
mon_bind_addrvec=config.get('mon_bind_addrvec', True),
|
|
|
|
)
|
2019-11-11 20:30:59 +00:00
|
|
|
log.info('Monitor IPs: %s' % ctx.ceph[cluster_name].mons)
|
2019-11-08 22:39:00 +00:00
|
|
|
|
|
|
|
with contextutil.nested(
|
2019-11-11 15:27:16 +00:00
|
|
|
lambda: ceph_initial(),
|
|
|
|
lambda: normalize_hostnames(ctx=ctx),
|
2019-11-08 22:39:00 +00:00
|
|
|
lambda: download_ceph_daemon(ctx=ctx, config=config),
|
2019-11-11 20:30:59 +00:00
|
|
|
lambda: ceph_log(ctx=ctx, config=config),
|
|
|
|
lambda: ceph_crash(ctx=ctx, config=config),
|
|
|
|
lambda: ceph_bootstrap(ctx=ctx, config=config),
|
2019-11-11 15:27:16 +00:00
|
|
|
lambda: ceph_mons(ctx=ctx, config=config),
|
2019-11-08 22:39:00 +00:00
|
|
|
lambda: ceph_mgrs(ctx=ctx, config=config),
|
2019-11-11 15:27:16 +00:00
|
|
|
lambda: ceph_osds(ctx=ctx, config=config),
|
2019-11-11 23:14:17 +00:00
|
|
|
lambda: ceph_mdss(ctx=ctx, config=config),
|
2019-11-11 20:30:59 +00:00
|
|
|
lambda: distribute_config_and_admin_keyring(ctx=ctx, config=config),
|
2019-11-08 22:39:00 +00:00
|
|
|
):
|
2019-11-12 22:15:32 +00:00
|
|
|
ctx.managers[cluster_name] = CephManager(
|
|
|
|
ctx.ceph[cluster_name].bootstrap_remote,
|
|
|
|
ctx=ctx,
|
|
|
|
logger=log.getChild('ceph_manager.' + cluster_name),
|
|
|
|
cluster=cluster_name,
|
|
|
|
ceph_daemon=True,
|
|
|
|
)
|
|
|
|
|
2019-11-08 22:39:00 +00:00
|
|
|
try:
|
2019-11-11 15:27:16 +00:00
|
|
|
log.info('Setup complete, yielding')
|
2019-11-08 22:39:00 +00:00
|
|
|
yield
|
|
|
|
|
|
|
|
finally:
|
2019-11-11 15:27:16 +00:00
|
|
|
log.info('Teardown begin')
|
|
|
|
|