| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145 | from __future__ import absolute_importfrom __future__ import unicode_literalsimport operatorimport sysfrom threading import Threadfrom docker.errors import APIErrorfrom six.moves import _thread as threadfrom six.moves.queue import Emptyfrom six.moves.queue import Queuefrom compose.cli.signals import ShutdownExceptionfrom compose.utils import get_output_streamdef perform_operation(func, arg, callback, index):    try:        callback((index, func(arg)))    except Exception as e:        callback((index, e))def parallel_execute(objects, func, index_func, msg):    """For a given list of objects, call the callable passing in the first    object we give it.    """    objects = list(objects)    stream = get_output_stream(sys.stderr)    writer = ParallelStreamWriter(stream, msg)    q = setup_queue(writer, objects, func, index_func)    done = 0    errors = {}    while done < len(objects):        try:            msg_index, result = q.get(timeout=1)        except Empty:            continue        # See https://github.com/docker/compose/issues/189        except thread.error:            raise ShutdownException()        if isinstance(result, APIError):            errors[msg_index] = "error", result.explanation            writer.write(msg_index, 'error')        elif isinstance(result, Exception):            errors[msg_index] = "unexpected_exception", result        else:            writer.write(msg_index, 'done')        done += 1    if not errors:        return    stream.write("\n")    for msg_index, (result, error) in errors.items():        stream.write("ERROR: for {}  {} \n".format(msg_index, error))        if result == 'unexpected_exception':            raise errordef setup_queue(writer, objects, func, index_func):    for obj in objects:        writer.initialize(index_func(obj))    q = Queue()    # TODO: limit the number of threads #1828    for obj in objects:        t = Thread(            target=perform_operation,            args=(func, obj, q.put, index_func(obj)))        t.daemon = True        t.start()    return qclass ParallelStreamWriter(object):    """Write out messages for operations happening in parallel.    Each operation has it's own line, and ANSI code characters are used    to jump to the correct line, and write over the line.    """    def __init__(self, stream, msg):        self.stream = stream        self.msg = msg        self.lines = []    def initialize(self, obj_index):        self.lines.append(obj_index)        self.stream.write("{} {} ... \r\n".format(self.msg, obj_index))        self.stream.flush()    def write(self, obj_index, status):        position = self.lines.index(obj_index)        diff = len(self.lines) - position        # move up        self.stream.write("%c[%dA" % (27, diff))        # erase        self.stream.write("%c[2K\r" % 27)        self.stream.write("{} {} ... {}\r".format(self.msg, obj_index, status))        # move back down        self.stream.write("%c[%dB" % (27, diff))        self.stream.flush()def parallel_operation(containers, operation, options, message):    parallel_execute(        containers,        operator.methodcaller(operation, **options),        operator.attrgetter('name'),        message)def parallel_remove(containers, options):    stopped_containers = [c for c in containers if not c.is_running]    parallel_operation(stopped_containers, 'remove', options, 'Removing')def parallel_stop(containers, options):    parallel_operation(containers, 'stop', options, 'Stopping')def parallel_start(containers, options):    parallel_operation(containers, 'start', options, 'Starting')def parallel_pause(containers, options):    parallel_operation(containers, 'pause', options, 'Pausing')def parallel_unpause(containers, options):    parallel_operation(containers, 'unpause', options, 'Unpausing')def parallel_kill(containers, options):    parallel_operation(containers, 'kill', options, 'Killing')def parallel_restart(containers, options):    parallel_operation(containers, 'restart', options, 'Restarting')
 |