2013-10-12 08:28:27 +00:00
|
|
|
"""
|
|
|
|
Handle parallel execution on remote hosts
|
|
|
|
"""
|
2012-11-14 16:11:39 +00:00
|
|
|
import logging
|
|
|
|
|
|
|
|
from teuthology import misc as teuthology
|
|
|
|
from teuthology.parallel import parallel
|
|
|
|
from teuthology.orchestra import run as tor
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
2012-12-27 23:33:07 +00:00
|
|
|
from gevent import queue as queue
|
|
|
|
from gevent import event as event
|
|
|
|
|
|
|
|
def _init_barrier(barrier_queue, remote):
|
2013-10-12 08:28:27 +00:00
|
|
|
"""current just queues a remote host"""
|
2012-12-27 23:33:07 +00:00
|
|
|
barrier_queue.put(remote)
|
|
|
|
|
|
|
|
def _do_barrier(barrier, barrier_queue, remote):
|
2013-10-12 08:28:27 +00:00
|
|
|
"""special case for barrier"""
|
2012-12-27 23:33:07 +00:00
|
|
|
barrier_queue.get()
|
|
|
|
if barrier_queue.empty():
|
|
|
|
barrier.set()
|
|
|
|
barrier.clear()
|
|
|
|
else:
|
|
|
|
barrier.wait()
|
|
|
|
|
|
|
|
barrier_queue.put(remote)
|
|
|
|
if barrier_queue.full():
|
|
|
|
barrier.set()
|
|
|
|
barrier.clear()
|
|
|
|
else:
|
|
|
|
barrier.wait()
|
|
|
|
|
2013-02-25 17:59:21 +00:00
|
|
|
def _exec_host(barrier, barrier_queue, remote, sudo, testdir, ls):
|
2013-10-12 08:28:27 +00:00
|
|
|
"""Execute command remotely"""
|
2012-12-14 17:30:15 +00:00
|
|
|
log.info('Running commands on host %s', remote.name)
|
2013-02-25 17:59:21 +00:00
|
|
|
args = [
|
|
|
|
'TESTDIR={tdir}'.format(tdir=testdir),
|
|
|
|
'bash',
|
|
|
|
'-s'
|
|
|
|
]
|
2012-11-14 16:11:39 +00:00
|
|
|
if sudo:
|
|
|
|
args.insert(0, 'sudo')
|
2013-02-25 17:59:21 +00:00
|
|
|
|
2012-11-20 15:53:52 +00:00
|
|
|
r = remote.run( args=args, stdin=tor.PIPE, wait=False)
|
2012-11-14 16:11:39 +00:00
|
|
|
r.stdin.writelines(['set -e\n'])
|
|
|
|
r.stdin.flush()
|
|
|
|
for l in ls:
|
2013-03-10 02:07:06 +00:00
|
|
|
l.replace('$TESTDIR', testdir)
|
2012-12-27 23:33:07 +00:00
|
|
|
if l == "barrier":
|
|
|
|
_do_barrier(barrier, barrier_queue, remote)
|
|
|
|
continue
|
|
|
|
|
2012-11-14 16:11:39 +00:00
|
|
|
r.stdin.writelines([l, '\n'])
|
|
|
|
r.stdin.flush()
|
|
|
|
r.stdin.writelines(['\n'])
|
|
|
|
r.stdin.flush()
|
|
|
|
r.stdin.close()
|
|
|
|
tor.wait([r])
|
|
|
|
|
2012-12-27 23:33:07 +00:00
|
|
|
def _generate_remotes(ctx, config):
|
2013-10-12 08:28:27 +00:00
|
|
|
"""Return remote roles and the type of role specified in config"""
|
2012-12-27 23:33:07 +00:00
|
|
|
if 'all' in config and len(config) == 1:
|
|
|
|
ls = config['all']
|
|
|
|
for remote in ctx.cluster.remotes.iterkeys():
|
|
|
|
yield (remote, ls)
|
|
|
|
elif 'clients' in config:
|
|
|
|
ls = config['clients']
|
|
|
|
for role in teuthology.all_roles_of_type(ctx.cluster, 'client'):
|
|
|
|
(remote,) = ctx.cluster.only('client.{r}'.format(r=role)).remotes.iterkeys()
|
|
|
|
yield (remote, ls)
|
|
|
|
del config['clients']
|
|
|
|
for role, ls in config.iteritems():
|
|
|
|
(remote,) = ctx.cluster.only(role).remotes.iterkeys()
|
|
|
|
yield (remote, ls)
|
|
|
|
else:
|
|
|
|
for role, ls in config.iteritems():
|
|
|
|
(remote,) = ctx.cluster.only(role).remotes.iterkeys()
|
|
|
|
yield (remote, ls)
|
|
|
|
|
2012-11-14 16:11:39 +00:00
|
|
|
def task(ctx, config):
|
|
|
|
"""
|
2012-12-27 23:33:07 +00:00
|
|
|
Execute commands on multiple hosts in parallel
|
2012-11-14 16:11:39 +00:00
|
|
|
|
|
|
|
tasks:
|
|
|
|
- ceph:
|
|
|
|
- ceph-fuse: [client.0, client.1]
|
|
|
|
- pexec:
|
|
|
|
client.0:
|
|
|
|
- while true; do echo foo >> bar; done
|
|
|
|
client.1:
|
|
|
|
- sleep 1
|
|
|
|
- tail -f bar
|
|
|
|
- interactive:
|
|
|
|
|
2012-12-27 23:33:07 +00:00
|
|
|
Execute commands on all hosts in the cluster in parallel. This
|
|
|
|
is useful if there are many hosts and you want to run the same
|
|
|
|
command on all:
|
|
|
|
|
|
|
|
tasks:
|
|
|
|
- pexec:
|
|
|
|
all:
|
2013-02-18 20:14:12 +00:00
|
|
|
- grep FAIL /var/log/ceph/*
|
2012-12-27 23:33:07 +00:00
|
|
|
|
|
|
|
Or if you want to run in parallel on all clients:
|
|
|
|
|
|
|
|
tasks:
|
|
|
|
- pexec:
|
|
|
|
clients:
|
2013-01-23 20:37:39 +00:00
|
|
|
- dd if=/dev/zero of={testdir}/mnt.* count=1024 bs=1024
|
2012-12-27 23:33:07 +00:00
|
|
|
|
|
|
|
You can also ensure that parallel commands are synchronized with the
|
|
|
|
special 'barrier' statement:
|
|
|
|
|
|
|
|
tasks:
|
|
|
|
- pexec:
|
|
|
|
clients:
|
2013-01-23 20:37:39 +00:00
|
|
|
- cd {testdir}/mnt.*
|
2012-12-27 23:33:07 +00:00
|
|
|
- while true; do
|
|
|
|
- barrier
|
|
|
|
- dd if=/dev/zero of=./foo count=1024 bs=1024
|
|
|
|
- done
|
|
|
|
|
|
|
|
The above writes to the file foo on all clients over and over, but ensures that
|
|
|
|
all clients perform each write command in sync. If one client takes longer to
|
|
|
|
write, all the other clients will wait.
|
|
|
|
|
2012-11-14 16:11:39 +00:00
|
|
|
"""
|
|
|
|
log.info('Executing custom commands...')
|
|
|
|
assert isinstance(config, dict), "task pexec got invalid config"
|
|
|
|
|
|
|
|
sudo = False
|
|
|
|
if 'sudo' in config:
|
|
|
|
sudo = config['sudo']
|
|
|
|
del config['sudo']
|
|
|
|
|
2013-02-25 17:59:21 +00:00
|
|
|
testdir = teuthology.get_testdir(ctx)
|
2012-12-27 23:33:07 +00:00
|
|
|
|
|
|
|
remotes = list(_generate_remotes(ctx, config))
|
|
|
|
count = len(remotes)
|
|
|
|
barrier_queue = queue.Queue(count)
|
|
|
|
barrier = event.Event()
|
|
|
|
|
|
|
|
for remote in remotes:
|
|
|
|
_init_barrier(barrier_queue, remote[0])
|
|
|
|
with parallel() as p:
|
|
|
|
for remote in remotes:
|
2013-02-25 17:59:21 +00:00
|
|
|
p.spawn(_exec_host, barrier, barrier_queue, remote[0], sudo, testdir, remote[1])
|