Как периодически планировать некоторые задачи в django, используя бит сельдерея django?

голоса
0

Это моя первая попытка с сельдереем, так что, очевидно, есть некоторые проблемы. Здесь я хочу создавать Task Objectкаждые 1 минуту в базе данных, поэтому я использовал django-celery-beat. Позже я хочу использовать пользовательский класс планировщика, поэтому для этой цели я использовал пакет django-celery-beat Но я не получаю результатов.

Я использовал джанго-форму для создания объекта TaskModel и написания задачи в файле tasks.py, чтобы запускать представление каждые 1 минуту. Но это бросает эту ошибку

Exception Type: EncodeError
Exception Value:    
Object of type timedelta is not JSON serializable

Я запустил сельдерей в одной консоли с помощью команды $ celery -A celery_demo beat -l info --scheduler django_celery_beat.schedulers:DatabaseSchedulerи запустил сервер django в другой консоли с помощью команды py manage.py runserver

settings.py

CELERY_BROKER_URL = 'amqp://localhost'

CELERY_BEAT_SCHEDULE = {

        'task-first': {
        'task': 'scheduler.tasks.create_task',
        'schedule': timedelta(minutes=1)
       },

сельдерей.пи

from __future__ import absolute_import, unicode_literals

import os

from celery import Celery

# set the default Django settings module for the 'celery' program.
from django.conf import settings

os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'celery_demo.settings')

app = Celery('celery_demo')

# Using a string here means the worker doesn't have to serialize
# the configuration object to child processes.
# - namespace='CELERY' means all celery-related configuration keys
#   should have a `CELERY_` prefix.
app.config_from_object('django.conf:settings')

# Load task modules from all registered Django app configs.
app.autodiscover_tasks(settings.INSTALLED_APPS)


@app.task(bind=True)
def debug_task(self):
    print('Request: {0!r}'.format(self.request))

init.py

from __future__ import absolute_import, unicode_literals

# This will make sure the app is always imported when
# Django starts so that shared_task will use this app.
from .celery import app as celery_app

__all__ = ('celery_app',)

приложение/задачи.пи

from celery.task import task
from django.shortcuts import redirect, render

from scheduler.models import Task

@task(name='create_task')
def add_task_celery(name=None, date=None, frequency=None):
    Task.objects.create(name=name, date=date, frequency=frequency)
    return redirect('list_tasks')

appp/views.py

def add_task(request):
    form = AddTaskForm()
    if request.method == 'POST':
        form = AddTaskForm(request.POST)
        if form.is_valid():
            name = form.cleaned_data.get('name')
            date = form.cleaned_data.get('date')
            freq = form.cleaned_data.get('frequency')

            add_task_celery.delay(name,date,freq)
    return render(request, 'add_task.html', {'form': form})

модели

id=до 6
Задан 01/06/2020 в 16:04
источник пользователем
На других языках...                            


1 ответов

голоса
0

Проблема заключается в третьем параметре в определении задачи, который freqи есть тип timedelta. https://docs.djangoproject.com/en/3.0/ref/models/fields/#durationfield

Это должно быть сериализовано перед переходом к задаче. Одним из простых способов будет.

1) Вы можете явно сериализовать это поле и передать, в задаче снова преобразовать его в объект timedelta.

2) Еще одна вещь, которую вы можете попробовать - это явная передача сериалайзера в параметрах.

add_task_celery.delay(name,date,freq, serializer='json')

3) вы также можете установить значение для установки CELERY_TASK_SERIALIZER = 'json' (значение по умолчанию - pickle)

Ответил 04/06/2020 в 07:59
источник пользователем

Cookies help us deliver our services. By using our services, you agree to our use of cookies. Learn more