Django: Ako kaskádová aktualizácia cez viacero modelov?

0

Otázka

Ja som písať Django na základe žiadosti sledovať objekty (Objekt) a ich údržby. Objekts môžu byť prepojené na miesto.

Umiestnenie (0/1) --- (n) Objekt (1) --- (n) Úloha

Miesto, Objekt a Úlohy všetci mať stav v oblasti nasledovné hodnoty:

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

Chcem, aby bolo Miesto na mape zmení farbu na základe stavu súvisiace Objekts a ultimatelly Úlohy.

Snažil som sa sledovať django najlepšie postupy a vytvoriť tuku 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 nejako nie som si istý, že o moju koncepciu tu... Ako môže aktualizáciu na Úlohu získavať aktualizácie na súvisiace Objekt. A ako by sa Objekt ďalšie spustenie aktualizácie na Mieste, - v prípade potreby na všetkých?

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

Najlepšiu odpoveď

1

Možným riešením je použiť signály. Ja som ho zaviedli, ako tieto, bez zeler teraz:

# 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ý setup som urobil na Mieste model, ktorý tiež reaguje na post_save signál zo Objekt. Nie som si istý, či je to najlepšie miesto na uloženie objekt_update_status() funkcia v signals.py súbor, ale snaží sa dať ho do models.py skončil s kruhovým chyba import.

2021-11-24 11:19:32

V iných jazykoch

Táto stránka je v iných jazykoch

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