2011-05-31 20:51:48 +00:00
|
|
|
from cStringIO import StringIO
|
|
|
|
|
|
|
|
import os
|
|
|
|
import logging
|
|
|
|
import configobj
|
|
|
|
import time
|
2011-06-02 16:09:08 +00:00
|
|
|
import urllib2
|
|
|
|
import urlparse
|
2011-05-31 20:51:48 +00:00
|
|
|
|
2011-06-16 01:06:57 +00:00
|
|
|
from orchestra import run
|
|
|
|
|
2011-05-31 20:51:48 +00:00
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
2011-06-10 00:05:55 +00:00
|
|
|
def get_ceph_binary_url(branch=None, tag=None, sha1=None, flavor=None):
|
2011-06-10 18:12:34 +00:00
|
|
|
if flavor is None:
|
|
|
|
flavor = ''
|
|
|
|
else:
|
|
|
|
# TODO hardcoding amd64 here for simplicity; clients will try
|
|
|
|
# to fetch the tarball matching their arch, non-x86_64 just
|
|
|
|
# won't find anything and the test will fail. trying to
|
|
|
|
# support cross-arch clusters is messy because nothing
|
|
|
|
# guarantees the same sha1 of "master" has been built for all
|
|
|
|
# of them. hoping for yagni.
|
|
|
|
flavor = '-{flavor}-amd64'.format(flavor=flavor)
|
|
|
|
BASE = 'http://ceph.newdream.net/gitbuilder{flavor}/output/'.format(flavor=flavor)
|
|
|
|
|
2011-06-10 00:05:55 +00:00
|
|
|
if sha1 is not None:
|
|
|
|
assert branch is None, "cannot set both sha1 and branch"
|
|
|
|
assert tag is None, "cannot set both sha1 and tag"
|
2011-06-09 21:08:45 +00:00
|
|
|
else:
|
2011-06-10 00:05:55 +00:00
|
|
|
# gitbuilder uses remote-style ref names for branches, mangled to
|
|
|
|
# have underscores instead of slashes; e.g. origin_master
|
|
|
|
if tag is not None:
|
|
|
|
ref = tag
|
|
|
|
assert branch is None, "cannot set both branch and tag"
|
|
|
|
else:
|
|
|
|
if branch is None:
|
|
|
|
branch = 'master'
|
|
|
|
ref = 'origin_{branch}'.format(branch=branch)
|
|
|
|
|
|
|
|
sha1_url = urlparse.urljoin(BASE, 'ref/{ref}/sha1'.format(ref=ref))
|
|
|
|
log.debug('Translating ref to sha1 using url %s', sha1_url)
|
|
|
|
sha1_fp = urllib2.urlopen(sha1_url)
|
|
|
|
sha1 = sha1_fp.read().rstrip('\n')
|
|
|
|
sha1_fp.close()
|
2011-06-09 22:43:43 +00:00
|
|
|
|
|
|
|
log.debug('Using ceph sha1 %s', sha1)
|
2011-06-02 16:09:08 +00:00
|
|
|
bindir_url = urlparse.urljoin(BASE, 'sha1/{sha1}/'.format(sha1=sha1))
|
2011-06-09 21:02:44 +00:00
|
|
|
return (sha1, bindir_url)
|
2011-05-31 20:51:48 +00:00
|
|
|
|
|
|
|
def feed_many_stdins(fp, processes):
|
|
|
|
while True:
|
|
|
|
data = fp.read(8192)
|
|
|
|
if not data:
|
|
|
|
break
|
|
|
|
for proc in processes:
|
|
|
|
proc.stdin.write(data)
|
|
|
|
|
|
|
|
def feed_many_stdins_and_close(fp, processes):
|
|
|
|
feed_many_stdins(fp, processes)
|
|
|
|
for proc in processes:
|
|
|
|
proc.stdin.close()
|
|
|
|
|
|
|
|
def get_mons(roles, ips):
|
|
|
|
mons = {}
|
|
|
|
for idx, roles in enumerate(roles):
|
|
|
|
for role in roles:
|
|
|
|
if not role.startswith('mon.'):
|
|
|
|
continue
|
|
|
|
mon_id = int(role[len('mon.'):])
|
|
|
|
addr = '{ip}:{port}'.format(
|
|
|
|
ip=ips[idx],
|
|
|
|
port=6789+mon_id,
|
|
|
|
)
|
|
|
|
mons[role] = addr
|
|
|
|
assert mons
|
|
|
|
return mons
|
|
|
|
|
|
|
|
def generate_caps(type_):
|
|
|
|
defaults = dict(
|
|
|
|
osd=dict(
|
|
|
|
mon='allow *',
|
|
|
|
osd='allow *',
|
|
|
|
),
|
|
|
|
mds=dict(
|
|
|
|
mon='allow *',
|
|
|
|
osd='allow *',
|
|
|
|
mds='allow',
|
|
|
|
),
|
|
|
|
client=dict(
|
|
|
|
mon='allow rw',
|
|
|
|
osd='allow rwx pool=data,rbd',
|
|
|
|
mds='allow',
|
|
|
|
),
|
|
|
|
)
|
|
|
|
for subsystem, capability in defaults[type_].items():
|
|
|
|
yield '--cap'
|
|
|
|
yield subsystem
|
|
|
|
yield capability
|
|
|
|
|
|
|
|
def skeleton_config(roles, ips):
|
|
|
|
"""
|
|
|
|
Returns a ConfigObj that's prefilled with a skeleton config.
|
|
|
|
|
|
|
|
Use conf[section][key]=value or conf.merge to change it.
|
|
|
|
|
|
|
|
Use conf.write to write it out, override .filename first if you want.
|
|
|
|
"""
|
|
|
|
path = os.path.join(os.path.dirname(__file__), 'ceph.conf')
|
|
|
|
conf = configobj.ConfigObj(path, file_error=True)
|
|
|
|
mons = get_mons(roles=roles, ips=ips)
|
|
|
|
for role, addr in mons.iteritems():
|
|
|
|
conf.setdefault(role, {})
|
|
|
|
conf[role]['mon addr'] = addr
|
|
|
|
return conf
|
|
|
|
|
|
|
|
def roles_of_type(roles_for_host, type_):
|
|
|
|
prefix = '{type}.'.format(type=type_)
|
|
|
|
for name in roles_for_host:
|
|
|
|
if not name.startswith(prefix):
|
|
|
|
continue
|
|
|
|
id_ = name[len(prefix):]
|
|
|
|
yield id_
|
|
|
|
|
2011-06-01 23:04:52 +00:00
|
|
|
def is_type(type_):
|
|
|
|
"""
|
|
|
|
Returns a matcher function for whether role is of type given.
|
|
|
|
"""
|
|
|
|
prefix = '{type}.'.format(type=type_)
|
|
|
|
def _is_type(role):
|
|
|
|
return role.startswith(prefix)
|
|
|
|
return _is_type
|
|
|
|
|
2011-06-03 21:47:44 +00:00
|
|
|
def num_instances_of_type(cluster, type_):
|
|
|
|
remotes_and_roles = cluster.remotes.items()
|
|
|
|
roles = [roles for (remote, roles) in remotes_and_roles]
|
2011-05-31 20:51:48 +00:00
|
|
|
prefix = '{type}.'.format(type=type_)
|
|
|
|
num = sum(sum(1 for role in hostroles if role.startswith(prefix)) for hostroles in roles)
|
|
|
|
return num
|
|
|
|
|
2011-06-01 23:04:52 +00:00
|
|
|
def create_simple_monmap(remote, conf):
|
2011-05-31 20:51:48 +00:00
|
|
|
"""
|
|
|
|
Writes a simple monmap based on current ceph.conf into <tmpdir>/monmap.
|
|
|
|
|
|
|
|
Assumes ceph_conf is up to date.
|
|
|
|
|
|
|
|
Assumes mon sections are named "mon.*", with the dot.
|
|
|
|
"""
|
|
|
|
def gen_addresses():
|
|
|
|
for section, data in conf.iteritems():
|
|
|
|
PREFIX = 'mon.'
|
|
|
|
if not section.startswith(PREFIX):
|
|
|
|
continue
|
|
|
|
name = section[len(PREFIX):]
|
|
|
|
addr = data['mon addr']
|
|
|
|
yield (name, addr)
|
|
|
|
|
|
|
|
addresses = list(gen_addresses())
|
|
|
|
assert addresses, "There are no monitors in config!"
|
|
|
|
log.debug('Ceph mon addresses: %s', addresses)
|
|
|
|
|
|
|
|
args = [
|
2011-06-07 18:45:29 +00:00
|
|
|
'/tmp/cephtest/binary/usr/local/bin/ceph-coverage',
|
2011-06-10 18:17:11 +00:00
|
|
|
'/tmp/cephtest/archive/coverage',
|
2011-05-31 20:51:48 +00:00
|
|
|
'/tmp/cephtest/binary/usr/local/bin/monmaptool',
|
|
|
|
'--create',
|
|
|
|
'--clobber',
|
|
|
|
]
|
|
|
|
for (name, addr) in addresses:
|
|
|
|
args.extend(('--add', name, addr))
|
|
|
|
args.extend([
|
|
|
|
'--print',
|
|
|
|
'/tmp/cephtest/monmap',
|
|
|
|
])
|
2011-06-01 23:04:52 +00:00
|
|
|
remote.run(
|
2011-05-31 20:51:48 +00:00
|
|
|
args=args,
|
|
|
|
)
|
|
|
|
|
2011-06-01 23:04:52 +00:00
|
|
|
def write_file(remote, path, data):
|
|
|
|
remote.run(
|
2011-05-31 20:51:48 +00:00
|
|
|
args=[
|
|
|
|
'python',
|
|
|
|
'-c',
|
|
|
|
'import shutil, sys; shutil.copyfileobj(sys.stdin, file(sys.argv[1], "wb"))',
|
|
|
|
path,
|
|
|
|
],
|
|
|
|
stdin=data,
|
|
|
|
)
|
|
|
|
|
2011-06-01 23:04:52 +00:00
|
|
|
def get_file(remote, path):
|
2011-05-31 20:51:48 +00:00
|
|
|
"""
|
|
|
|
Read a file from remote host into memory.
|
|
|
|
"""
|
2011-06-01 23:04:52 +00:00
|
|
|
proc = remote.run(
|
2011-05-31 20:51:48 +00:00
|
|
|
args=[
|
|
|
|
'cat',
|
|
|
|
'--',
|
|
|
|
path,
|
|
|
|
],
|
|
|
|
stdout=StringIO(),
|
|
|
|
)
|
|
|
|
data = proc.stdout.getvalue()
|
|
|
|
return data
|
|
|
|
|
2011-06-01 23:04:52 +00:00
|
|
|
def wait_until_healthy(remote):
|
2011-05-31 20:51:48 +00:00
|
|
|
"""Wait until a Ceph cluster is healthy."""
|
|
|
|
while True:
|
2011-06-01 23:04:52 +00:00
|
|
|
r = remote.run(
|
2011-05-31 20:51:48 +00:00
|
|
|
args=[
|
2011-06-07 18:45:29 +00:00
|
|
|
'/tmp/cephtest/binary/usr/local/bin/ceph-coverage',
|
2011-06-10 18:17:11 +00:00
|
|
|
'/tmp/cephtest/archive/coverage',
|
2011-05-31 20:51:48 +00:00
|
|
|
'/tmp/cephtest/binary/usr/local/bin/ceph',
|
|
|
|
'-c', '/tmp/cephtest/ceph.conf',
|
|
|
|
'health',
|
|
|
|
'--concise',
|
|
|
|
],
|
|
|
|
stdout=StringIO(),
|
|
|
|
logger=log.getChild('health'),
|
|
|
|
)
|
|
|
|
out = r.stdout.getvalue()
|
|
|
|
log.debug('Ceph health: %s', out.rstrip('\n'))
|
|
|
|
if out.split(None, 1)[0] == 'HEALTH_OK':
|
|
|
|
break
|
|
|
|
time.sleep(1)
|
|
|
|
|
2011-06-01 23:04:52 +00:00
|
|
|
def wait_until_fuse_mounted(remote, fuse, mountpoint):
|
2011-05-31 20:51:48 +00:00
|
|
|
while True:
|
2011-06-01 23:04:52 +00:00
|
|
|
proc = remote.run(
|
2011-05-31 20:51:48 +00:00
|
|
|
args=[
|
|
|
|
'stat',
|
|
|
|
'--file-system',
|
|
|
|
'--printf=%T\n',
|
|
|
|
'--',
|
|
|
|
mountpoint,
|
|
|
|
],
|
|
|
|
stdout=StringIO(),
|
|
|
|
)
|
|
|
|
fstype = proc.stdout.getvalue().rstrip('\n')
|
|
|
|
if fstype == 'fuseblk':
|
|
|
|
break
|
|
|
|
log.debug('cfuse not yet mounted, got fs type {fstype!r}'.format(fstype=fstype))
|
|
|
|
|
|
|
|
# it shouldn't have exited yet; exposes some trivial problems
|
|
|
|
assert not fuse.exitstatus.ready()
|
|
|
|
|
|
|
|
time.sleep(5)
|
|
|
|
log.info('cfuse is mounted on %s', mountpoint)
|
2011-06-16 01:06:57 +00:00
|
|
|
|
|
|
|
def write_secret_file(remote, role, filename):
|
|
|
|
remote.run(
|
|
|
|
args=[
|
|
|
|
'/tmp/cephtest/binary/usr/local/bin/ceph-coverage',
|
|
|
|
'/tmp/cephtest/archive/coverage',
|
|
|
|
'/tmp/cephtest/binary/usr/local/bin/cauthtool',
|
|
|
|
'--name={role}'.format(role=role),
|
|
|
|
'--print-key',
|
|
|
|
'/tmp/cephtest/data/{role}.keyring'.format(role=role),
|
|
|
|
run.Raw('>'),
|
|
|
|
filename,
|
|
|
|
],
|
|
|
|
)
|