RabbitMQ 中的更新队列

时间:2020-12-23 12:21:54

标签: django rabbitmq celery

我有一个 task 要在对象的 POSTPATCH 之后执行,比如说 ABC

代码如下:

models.py

class ABC(models.Model):
    start_at = models.DateTimeField()

task.py

@app.task
def do_something(obj_id):
    try:
        abc = ABC.objects.get(id=obj_id)
    except ObjectDoesNotExist:
        return

    eta = settings.ETA
    do_something.apply_async([abc.id], eta=eta)
    return

views.py

class ABCPost(CreateAPIView):
    serializer_class = ABCSerializer

    def post(self, request):
        # create object
        # call task do_something
    
    def patch(self, request):
        # update the `start_at` field for ABC
        # call task do_something

因此,当字段更新时,排队的消息应该在更新后的 start_at 值处执行。但是,使用上面的代码,两条消息排队等待相同的对象,但时间戳不同。我怎样才能避免这种情况?

提前致谢

1 个答案:

答案 0 :(得分:1)

当您创建对象时,您需要存储 task_id :

abc = ABC.objects.get(pk=PK)
abc.task_id = do_something.apply_async([abc.pk], eta=eta).id
abc.save(updated_fields=['task_id'])

将此字段添加到 ABC 模型中,并将其存储在其中。

当你在补丁中时,你需要撤销第一个任务,如果还没有执行:

from celery.result import AsyncResult
abc = ABC.objects.get(pk=PK)
AsyncResult(abc.task_id).revoke()
abc.task_id = do_something.apply_async([abc.pk], eta=eta).id
abc.save(updated_fields=['task_id'])

这样你的队列中仍然会有 2 条消息,但第一条在执行时会被跳过(撤销并没有从队列中移除,只是在弹出时不执行)

相关问题