Django: Jak cascade update přes více modelů?

0

Otázka

Píšu Django založené aplikace pro sledování objektů (Objekt) a jejich úkoly údržby. Objekts mohou být spojeny k umístění.

Umístění (0/1) --- (n) Objekt (1) --- (n) Úkol

Místo, Objekt a Úkolem všechny mají status pole s následujícími hodnotami:

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

Chci, aby Umístění značka na mapě změní barvu na základě stavu souvisejících Objekts a ultimatelly Úkoly.

Snažil jsem se sledovat django osvědčených postupů a vytvořit fat 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 nějak si nejsem jistý, o mé koncepci, Jak můžete aktualizaci na Úkol vyvolat aktualizaci na související Objekt. A jak bude Objekt dále spustí aktualizace na Umístění - pokud je to vůbec nutné?

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

Nejlepší odpověď

1

Možné řešení je použít signály. Implementoval jsem to takto, bez celeru pro teď:

# 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é nastavení jsem udělal na Umístění modelu, který také reaguje na post_save signál z Objekt. Nejsem si jistý, jestli je to nejlepší místo pro uložení objekt_update_status() funkce v signals.py souboru, ale snaží se dát to do models.py skončil s kruhovým chyby při importu.

2021-11-24 11:19:32

V jiných jazycích

Tato stránka je v jiných jazycích

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