| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103 | # Example::#    >>> R = A.apply_async()#    >>> list(joinall(R))#    [['A 0', 'A 1', 'A 2', 'A 3', 'A 4', 'A 5', 'A 6', 'A 7', 'A 8', 'A 9'],#    ['B 0', 'B 1', 'B 2', 'B 3', 'B 4', 'B 5', 'B 6', 'B 7', 'B 8', 'B 9'],#    ['C 0', 'C 1', 'C 2', 'C 3', 'C 4', 'C 5', 'C 6', 'C 7', 'C 8', 'C 9'],#    ['D 0', 'D 1', 'D 2', 'D 3', 'D 4', 'D 5', 'D 6', 'D 7', 'D 8', 'D 9'],#    ['E 0', 'E 1', 'E 2', 'E 3', 'E 4', 'E 5', 'E 6', 'E 7', 'E 8', 'E 9'],#    ['F 0', 'F 1', 'F 2', 'F 3', 'F 4', 'F 5', 'F 6', 'F 7', 'F 8', 'F 9'],#    ['G 0', 'G 1', 'G 2', 'G 3', 'G 4', 'G 5', 'G 6', 'G 7', 'G 8', 'G 9'],#    ['H 0', 'H 1', 'H 2', 'H 3', 'H 4', 'H 5', 'H 6', 'H 7', 'H 8', 'H 9']]### Joining the graph asynchronously with a callback# (Note: only two levels, the deps are considered final#        when the second task is ready.)##    >>> unlock_graph.apply_async((A.apply_async(),#    ...                           A_callback.s()), countdown=1)from celery import chord, group, task, signature, uuidfrom celery.result import AsyncResult, ResultSet, allow_join_resultfrom collections import deque@task()def add(x, y):    return x + y@task()def make_request(id, url):    print('GET {0!r}'.format(url))    return url@task()def B_callback(urls, id):    print('batch {0} done'.format(id))    return urls@task()def B(id):    return chord(        make_request.s(id, '{0} {1!r}'.format(id, i))        for i in range(10)    )(B_callback.s(id))@task()def A():    return group(B.s(c) for c in 'ABCDEFGH').apply_async()def joinall(R, timeout=None, propagate=True):    stack = deque([R])    try:        use_native = joinall.backend.supports_native_join    except AttributeError:        use_native = False    while stack:        res = stack.popleft()        if isinstance(res, ResultSet):            j = res.join_native if use_native else res.join            stack.extend(j(timeout=timeout, propagate=propagate))        elif isinstance(res, AsyncResult):            stack.append(res.get(timeout=timeout, propagate=propagate))        else:            yield res@task()def unlock_graph(result, callback,                 interval=1, propagate=False, max_retries=None):    if result.ready():        second_level_res = result.get()        if second_level_res.ready():            with allow_join_result():                signature(callback).delay(list(joinall(                    second_level_res, propagate=propagate)))    else:        unlock_graph.retry(countdown=interval, max_retries=max_retries)@task()def A_callback(res):    print('Everything is done: {0!r}'.format(res))    return resclass chord2(object):    def __init__(self, tasks, **options):        self.tasks = tasks        self.options = options    def __call__(self, body, **options):        body.options.setdefault('task_id', uuid())        unlock_graph.apply_async()
 |