Celery: Callback after task hierarchy

后端 未结 2 978
误落风尘
误落风尘 2021-01-12 02:39

I\'m using Celery from a webapp to start a task hierarchy.

Tasks

I\'m using the following tasks:

  • task_a
  • task_b
相关标签:
2条回答
  • 2021-01-12 03:07

    The solution turned out to be the dynamic task feature provided in this pull request: https://github.com/celery/celery/pull/817. With this, each task can return a group of subtasks, which will then replace the original taks in the queue.

    0 讨论(0)
  • 2021-01-12 03:15

    Suppose you have these tasks:

    celery = Celery(
        broker="amqp://test:test@localhost:5672/test"
    )
    celery.conf.update(
        CELERY_RESULT_BACKEND = "mongodb",
    )
    
    
    @celery.task
    def task_a(result):
        print 'task_a:', result
        return result
    
    @celery.task
    def task_b(result):
        print 'task_b:', result
        return result
    
    @celery.task
    def task_c(result):
        print 'task_c:', result
        return result
    
    @celery.task
    def notify_user(result):
        print result
        return result
    

    For a given input data (as you drawn it):

        tree = [
            [["C1", "C2", "C3"], ["C4", "C5"]], [["C6", "C7", "C8"], ["C9"]]
        ]
    

    You can do:

        a_group = []
        for ia, a in enumerate(tree):
            print "A%s:" % ia
            b_group = []
            for ib, b in enumerate(a):
                print " - B%s:" % ib
                for c in b:
                    print '   -', c
    
                c_group = group([task_c.s(c) for c in b])
    
                b_group.append(c_group | task_b.s())
    
            a_group.append(group(b_group) | task_a.s())
    
        final_task = group(a_group) | notify_user.s()
    

    It's representation is (don't read it, it's ugly :)

    [[[__main__.task_c('C1'), __main__.task_c('C2'), __main__.task_c('C3')] | __main__.task_b(), [__main__.task_c('C4'), __main__.task_c('C5')] | __main__.task_b()] | __main__.task_a(), [[__main__.task_c('C6'), __main__.task_c('C7'), __main__.task_c('C8')] | __main__.task_b(), [__main__.task_c('C9')] | __main__.task_b()] | __main__.task_a()] | __main__.notify_user()
    

    And the data passed into notify_user would be:

    [[['C1', 'C2', 'C3'], ['C4', 'C5']], [['C6', 'C7', 'C8'], ['C9']]]
    

    Everything is run via callbacks (chords) so there are no tasks waiting for other tasks.

    0 讨论(0)
提交回复
热议问题