¿Cómo programar alguna tarea en django periódicamente usando el ritmo de apio de django?

votos
0

Este es mi primer intento con el apio, así que obviamente hay algunos problemas. Aquí quiero crear Task Objectcada 1 minuto en la base de datos, así que usédjango-celery-beat... Quiero usar la clase de programador personalizado más tarde así que para este propósito usé el paquete django-celery-beat Pero no estoy obteniendo los resultados.

Usé el formulario django para crear el objeto TaskModel y escribir una tarea en tasks.py para ejecutar la vista cada 1 minuto. Pero arroja este error

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

Inicié el apio en una consola con el comando $ celery -A celery_demo beat -l info --scheduler django_celery_beat.schedulers:DatabaseSchedulere inicié el servidor django en otra consola con el comando 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)
       },

apio.py

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',)

app/tasks.py

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})

modelos

id=pre-6
Publicado el 01/06/2020 a las 16:04
fuente por usuario
En otros idiomas...                            


1 respuestas

votos
0

El problema es con el tercer parámetro en la definición de la tarea, que es freqy es de tipo timedelta. https://docs.djangoproject.com/en/3.0/ref/models/fields/#durationfield

Esto debe ser serializado antes de pasar a la tarea. una forma simple sería.

1) puedes serializar explícitamente este campo y pasar, en la tarea de nuevo convertirlo en objeto timedelta.

2) otra cosa que puedes intentar es pasar el serializador en los parámetros explícitamente.

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

3) también puede establecer un valor para fijar CELERY_TASK_SERIALIZER = 'json' (el valor por defecto es pepinillo)

Respondida el 04/06/2020 a las 07:59
fuente por usuario

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