parallel.py 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. from __future__ import absolute_import
  2. from __future__ import unicode_literals
  3. import logging
  4. import operator
  5. import sys
  6. from threading import Thread
  7. from docker.errors import APIError
  8. from six.moves import _thread as thread
  9. from six.moves.queue import Empty
  10. from six.moves.queue import Queue
  11. from compose.cli.signals import ShutdownException
  12. from compose.utils import get_output_stream
  13. log = logging.getLogger(__name__)
  14. def parallel_execute(objects, func, get_name, msg, get_deps=None):
  15. """Runs func on objects in parallel while ensuring that func is
  16. ran on object only after it is ran on all its dependencies.
  17. get_deps called on object must return a collection with its dependencies.
  18. get_name called on object must return its name.
  19. """
  20. objects = list(objects)
  21. stream = get_output_stream(sys.stderr)
  22. writer = ParallelStreamWriter(stream, msg)
  23. for obj in objects:
  24. writer.initialize(get_name(obj))
  25. q = setup_queue(objects, func, get_deps, get_name)
  26. done = 0
  27. errors = {}
  28. results = []
  29. error_to_reraise = None
  30. while done < len(objects):
  31. try:
  32. obj, result, exception = q.get(timeout=1)
  33. except Empty:
  34. continue
  35. # See https://github.com/docker/compose/issues/189
  36. except thread.error:
  37. raise ShutdownException()
  38. if exception is None:
  39. writer.write(get_name(obj), 'done')
  40. results.append(result)
  41. elif isinstance(exception, APIError):
  42. errors[get_name(obj)] = exception.explanation
  43. writer.write(get_name(obj), 'error')
  44. else:
  45. errors[get_name(obj)] = exception
  46. error_to_reraise = exception
  47. done += 1
  48. for obj_name, error in errors.items():
  49. stream.write("\nERROR: for {} {}\n".format(obj_name, error))
  50. if error_to_reraise:
  51. raise error_to_reraise
  52. return results
  53. def _no_deps(x):
  54. return []
  55. def setup_queue(objects, func, get_deps, get_name):
  56. if get_deps is None:
  57. get_deps = _no_deps
  58. results = Queue()
  59. output = Queue()
  60. def consumer():
  61. started = set() # objects being processed
  62. finished = set() # objects which have been processed
  63. def ready(obj):
  64. """
  65. Returns true if obj is ready to be processed:
  66. - all dependencies have been processed
  67. - obj is not already being processed
  68. """
  69. return obj not in started and all(
  70. dep not in objects or dep in finished
  71. for dep in get_deps(obj)
  72. )
  73. while len(finished) < len(objects):
  74. for obj in filter(ready, objects):
  75. log.debug('Starting producer thread for {}'.format(obj))
  76. t = Thread(target=producer, args=(obj,))
  77. t.daemon = True
  78. t.start()
  79. started.add(obj)
  80. try:
  81. event = results.get(timeout=1)
  82. except Empty:
  83. continue
  84. obj = event[0]
  85. log.debug('Finished processing: {}'.format(obj))
  86. finished.add(obj)
  87. output.put(event)
  88. def producer(obj):
  89. try:
  90. result = func(obj)
  91. results.put((obj, result, None))
  92. except Exception as e:
  93. results.put((obj, None, e))
  94. t = Thread(target=consumer)
  95. t.daemon = True
  96. t.start()
  97. return output
  98. class ParallelStreamWriter(object):
  99. """Write out messages for operations happening in parallel.
  100. Each operation has it's own line, and ANSI code characters are used
  101. to jump to the correct line, and write over the line.
  102. """
  103. def __init__(self, stream, msg):
  104. self.stream = stream
  105. self.msg = msg
  106. self.lines = []
  107. def initialize(self, obj_index):
  108. if self.msg is None:
  109. return
  110. self.lines.append(obj_index)
  111. self.stream.write("{} {} ... \r\n".format(self.msg, obj_index))
  112. self.stream.flush()
  113. def write(self, obj_index, status):
  114. if self.msg is None:
  115. return
  116. position = self.lines.index(obj_index)
  117. diff = len(self.lines) - position
  118. # move up
  119. self.stream.write("%c[%dA" % (27, diff))
  120. # erase
  121. self.stream.write("%c[2K\r" % 27)
  122. self.stream.write("{} {} ... {}\r".format(self.msg, obj_index, status))
  123. # move back down
  124. self.stream.write("%c[%dB" % (27, diff))
  125. self.stream.flush()
  126. def parallel_operation(containers, operation, options, message):
  127. parallel_execute(
  128. containers,
  129. operator.methodcaller(operation, **options),
  130. operator.attrgetter('name'),
  131. message)
  132. def parallel_remove(containers, options):
  133. stopped_containers = [c for c in containers if not c.is_running]
  134. parallel_operation(stopped_containers, 'remove', options, 'Removing')
  135. def parallel_start(containers, options):
  136. parallel_operation(containers, 'start', options, 'Starting')
  137. def parallel_pause(containers, options):
  138. parallel_operation(containers, 'pause', options, 'Pausing')
  139. def parallel_unpause(containers, options):
  140. parallel_operation(containers, 'unpause', options, 'Unpausing')
  141. def parallel_kill(containers, options):
  142. parallel_operation(containers, 'kill', options, 'Killing')
  143. def parallel_restart(containers, options):
  144. parallel_operation(containers, 'restart', options, 'Restarting')