Django: Jak spadać kaskadą aktualizacja po kilku modeli?

0

Pytanie

Piszę aplikację w oparciu o Django, aby śledzić obiekty (obiekt) i ich zadania konserwacji. Obiekty mogą być związane z lokalizacją.

Lokalizacja (0/1) - - - (n) Obiekt (1) - - - (n) Zadanie

Lokalizacja, Obiekt i zadanie mają pole statusu z następującymi wartościami:

    RED = "red"
    YELLOW = "yellow"
    GREEN = "green"
    STATUS = [
        (RED, "Overdue tasks"),
        (YELLOW, "Pending tasks"),
        (GREEN, "All good"),
    ]

Chcę, aby znacznik na mapie lokalizacji zmieniał swój kolor w zależności od stanu powiązanych obiektów i ostatecznie Zadań.

Starałem się zgodnie z najlepszymi praktykami django i stworzyć grubą model.

from django.db import models
from locationapp.models import Location
from taskapp.models import Task
from rules.contrib.models import RulesModel

class Objekt(RulesModel):
    RED = "red"
    YELLOW = "yellow"
    GREEN = "green"
    STATUS = [
        (RED, "Overdue tasks"),
        (YELLOW, "Pending tasks"),
        (GREEN, "All good"),
    ]
    name = models.CharField(max_length=200)
    description = models.TextField(blank=True)
    location = models.ForeignKey(
        Location, on_delete=models.SET_NULL, null=True, blank=True
    )
    status = models.CharField(max_length=6, choices=STATUS, default=GREEN)

    def set_status(self):
        if Task.objects.filter(objekt=self.id).filter(status=Task.RED).exists():
            self.status = Objekt.RED
        elif Task.objects.filter(objekt=self.id).filter(status=Task.YELLOW).exists():
            self.status = Objekt.YELLOW
        else:
            self.status = Objekt.GREEN

Ale jakoś nie jestem pewien w swojej koncepcji tutaj... Jak aktualizacja zadania może spowodować aktualizację powiązanego obiektu. I jak Obiekt w przyszłości inicjuje aktualizacja Lokalizacji - jeśli to w ogóle konieczne?

django model python
2021-11-22 21:49:18
1

Najlepsza odpowiedź

1

Potencjalnym rozwiązaniem jest używanie sygnałów. Osiągnąłem to w następujący sposób, na razie bez selera:

# objektapp/apps.py
from django.apps import AppConfig


class ObjektappConfig(AppConfig):
    default_auto_field = 'django.db.models.BigAutoField'
    name = 'objektapp'

    def ready(self):
        import objektapp.signals
# objektapp/signals.py
from django.db.models.signals import post_save
from django.dispatch import receiver
from django.db import transaction
from taskapp.models import Task
from objektapp.models import Objekt

@receiver(post_save, sender=Task)
def set_status(sender, instance, created, **kwargs):
    # TODO: Use celery for async operation: https://docs.djangoproject.com/en/3.2/topics/db/transactions/
    transaction.on_commit(lambda: objekt_update_status(instance))

def objekt_update_status(task_instance):
    objekt = Objekt.objects.get(id=task_instance.objekt.id)

    new_objekt_status = Objekt.GREEN
    if Task.objects.filter(objekt=task_instance.objekt.id, status=Task.RED).exists():
        new_objekt_status = Objekt.RED
    elif Task.objects.filter(objekt=task_instance.objekt.id, status=Task.YELLOW).exists():
        new_objekt_status = Objekt.YELLOW

    if objekt.status != new_objekt_status:
        objekt.status = new_objekt_status
        objekt.save()

Podobną konfigurację zrobiłem dla modelu lokalizacji, która również reaguje na sygnał post_save od obiektu. Nie jestem pewien, czy to najlepsze miejsce do przechowywania funkcji object_update_status() w signals.py plik, ale staram się umieścić go w модели.ру ukończył z cyklicznej błędem importu.

2021-11-24 11:19:32

W innych językach

Ta strona jest w innych językach

Русский
..................................................................................................................
Italiano
..................................................................................................................
Română
..................................................................................................................
한국어
..................................................................................................................
हिन्दी
..................................................................................................................
Français
..................................................................................................................
Türk
..................................................................................................................
Česk
..................................................................................................................
Português
..................................................................................................................
ไทย
..................................................................................................................
中文
..................................................................................................................
Español
..................................................................................................................
Slovenský
..................................................................................................................