2016-04-08 18:46:13 +02:00
|
|
|
from __future__ import absolute_import
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
2018-01-04 21:16:46 +01:00
|
|
|
import unittest
|
2017-02-25 01:14:32 +01:00
|
|
|
from threading import Lock
|
|
|
|
|
2016-04-08 18:46:13 +02:00
|
|
|
import six
|
|
|
|
from docker.errors import APIError
|
|
|
|
|
2017-12-17 01:11:55 +01:00
|
|
|
from compose.parallel import GlobalLimit
|
2016-04-08 18:46:13 +02:00
|
|
|
from compose.parallel import parallel_execute
|
2016-04-11 14:03:35 +02:00
|
|
|
from compose.parallel import parallel_execute_iter
|
2017-08-10 01:46:47 +02:00
|
|
|
from compose.parallel import ParallelStreamWriter
|
2016-04-08 19:53:16 +02:00
|
|
|
from compose.parallel import UpstreamError
|
2016-04-08 18:46:13 +02:00
|
|
|
|
|
|
|
|
|
|
|
web = 'web'
|
|
|
|
db = 'db'
|
|
|
|
data_volume = 'data_volume'
|
|
|
|
cache = 'cache'
|
|
|
|
|
|
|
|
objects = [web, db, data_volume, cache]
|
|
|
|
|
|
|
|
deps = {
|
|
|
|
web: [db, cache],
|
|
|
|
db: [data_volume],
|
|
|
|
data_volume: [],
|
|
|
|
cache: [],
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-04-08 19:48:07 +02:00
|
|
|
def get_deps(obj):
|
2016-12-20 05:20:03 +01:00
|
|
|
return [(dep, None) for dep in deps[obj]]
|
2016-04-08 19:48:07 +02:00
|
|
|
|
|
|
|
|
2018-01-04 21:16:46 +01:00
|
|
|
class ParallelTest(unittest.TestCase):
|
|
|
|
|
|
|
|
def test_parallel_execute(self):
|
|
|
|
results, errors = parallel_execute(
|
|
|
|
objects=[1, 2, 3, 4, 5],
|
|
|
|
func=lambda x: x * 2,
|
|
|
|
get_name=six.text_type,
|
|
|
|
msg="Doubling",
|
|
|
|
)
|
|
|
|
|
|
|
|
assert sorted(results) == [2, 4, 6, 8, 10]
|
|
|
|
assert errors == {}
|
|
|
|
|
|
|
|
def test_parallel_execute_with_limit(self):
|
|
|
|
limit = 1
|
|
|
|
tasks = 20
|
|
|
|
lock = Lock()
|
|
|
|
|
|
|
|
def f(obj):
|
|
|
|
locked = lock.acquire(False)
|
|
|
|
# we should always get the lock because we're the only thread running
|
|
|
|
assert locked
|
|
|
|
lock.release()
|
|
|
|
return None
|
|
|
|
|
|
|
|
results, errors = parallel_execute(
|
|
|
|
objects=list(range(tasks)),
|
|
|
|
func=f,
|
|
|
|
get_name=six.text_type,
|
|
|
|
msg="Testing",
|
|
|
|
limit=limit,
|
|
|
|
)
|
|
|
|
|
|
|
|
assert results == tasks * [None]
|
|
|
|
assert errors == {}
|
|
|
|
|
|
|
|
def test_parallel_execute_with_global_limit(self):
|
|
|
|
GlobalLimit.set_global_limit(1)
|
|
|
|
self.addCleanup(GlobalLimit.set_global_limit, None)
|
|
|
|
tasks = 20
|
|
|
|
lock = Lock()
|
|
|
|
|
|
|
|
def f(obj):
|
|
|
|
locked = lock.acquire(False)
|
|
|
|
# we should always get the lock because we're the only thread running
|
|
|
|
assert locked
|
|
|
|
lock.release()
|
|
|
|
return None
|
|
|
|
|
|
|
|
results, errors = parallel_execute(
|
|
|
|
objects=list(range(tasks)),
|
|
|
|
func=f,
|
|
|
|
get_name=six.text_type,
|
|
|
|
msg="Testing",
|
|
|
|
)
|
|
|
|
|
|
|
|
assert results == tasks * [None]
|
|
|
|
assert errors == {}
|
|
|
|
|
|
|
|
def test_parallel_execute_with_deps(self):
|
|
|
|
log = []
|
|
|
|
|
|
|
|
def process(x):
|
|
|
|
log.append(x)
|
|
|
|
|
|
|
|
parallel_execute(
|
|
|
|
objects=objects,
|
|
|
|
func=process,
|
|
|
|
get_name=lambda obj: obj,
|
|
|
|
msg="Processing",
|
|
|
|
get_deps=get_deps,
|
|
|
|
)
|
|
|
|
|
|
|
|
assert sorted(log) == sorted(objects)
|
|
|
|
|
|
|
|
assert log.index(data_volume) < log.index(db)
|
|
|
|
assert log.index(db) < log.index(web)
|
|
|
|
assert log.index(cache) < log.index(web)
|
|
|
|
|
|
|
|
def test_parallel_execute_with_upstream_errors(self):
|
|
|
|
log = []
|
|
|
|
|
|
|
|
def process(x):
|
|
|
|
if x is data_volume:
|
|
|
|
raise APIError(None, None, "Something went wrong")
|
|
|
|
log.append(x)
|
|
|
|
|
|
|
|
parallel_execute(
|
|
|
|
objects=objects,
|
|
|
|
func=process,
|
|
|
|
get_name=lambda obj: obj,
|
|
|
|
msg="Processing",
|
|
|
|
get_deps=get_deps,
|
|
|
|
)
|
|
|
|
|
|
|
|
assert log == [cache]
|
|
|
|
|
|
|
|
events = [
|
|
|
|
(obj, result, type(exception))
|
|
|
|
for obj, result, exception
|
|
|
|
in parallel_execute_iter(objects, process, get_deps, None)
|
|
|
|
]
|
|
|
|
|
|
|
|
assert (cache, None, type(None)) in events
|
|
|
|
assert (data_volume, None, APIError) in events
|
|
|
|
assert (db, None, UpstreamError) in events
|
|
|
|
assert (web, None, UpstreamError) in events
|
2017-02-25 01:48:02 +01:00
|
|
|
|
|
|
|
|
|
|
|
def test_parallel_execute_alignment(capsys):
|
2018-03-05 14:28:46 +01:00
|
|
|
ParallelStreamWriter.instance = None
|
2017-02-25 01:48:02 +01:00
|
|
|
results, errors = parallel_execute(
|
|
|
|
objects=["short", "a very long name"],
|
|
|
|
func=lambda x: x,
|
|
|
|
get_name=six.text_type,
|
|
|
|
msg="Aligning",
|
|
|
|
)
|
|
|
|
|
|
|
|
assert errors == {}
|
|
|
|
|
|
|
|
_, err = capsys.readouterr()
|
|
|
|
a, b = err.split('\n')[:2]
|
|
|
|
assert a.index('...') == b.index('...')
|
2017-06-23 15:17:38 +02:00
|
|
|
|
|
|
|
|
2017-08-22 15:37:32 +02:00
|
|
|
def test_parallel_execute_ansi(capsys):
|
2018-03-05 14:28:46 +01:00
|
|
|
ParallelStreamWriter.instance = None
|
2017-08-22 15:37:32 +02:00
|
|
|
ParallelStreamWriter.set_noansi(value=False)
|
|
|
|
results, errors = parallel_execute(
|
|
|
|
objects=["something", "something more"],
|
|
|
|
func=lambda x: x,
|
|
|
|
get_name=six.text_type,
|
|
|
|
msg="Control characters",
|
|
|
|
)
|
|
|
|
|
|
|
|
assert errors == {}
|
|
|
|
|
|
|
|
_, err = capsys.readouterr()
|
|
|
|
assert "\x1b" in err
|
|
|
|
|
|
|
|
|
|
|
|
def test_parallel_execute_noansi(capsys):
|
2018-03-05 14:28:46 +01:00
|
|
|
ParallelStreamWriter.instance = None
|
2017-08-10 01:46:47 +02:00
|
|
|
ParallelStreamWriter.set_noansi()
|
2017-06-23 15:17:38 +02:00
|
|
|
results, errors = parallel_execute(
|
2017-08-22 15:37:32 +02:00
|
|
|
objects=["something", "something more"],
|
2017-06-23 15:17:38 +02:00
|
|
|
func=lambda x: x,
|
|
|
|
get_name=six.text_type,
|
2017-08-22 15:37:32 +02:00
|
|
|
msg="Control characters",
|
2017-06-23 15:17:38 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
assert errors == {}
|
|
|
|
|
|
|
|
_, err = capsys.readouterr()
|
2017-08-22 15:37:32 +02:00
|
|
|
assert "\x1b" not in err
|