Skip to content
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
204 changes: 175 additions & 29 deletions api/drf_views.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,13 +6,15 @@
Avg,
Case,
Count,
Exists,
ExpressionWrapper,
F,
OuterRef,
Prefetch,
Q,
Subquery,
Sum,
Value,
When,
)
from django.db.models.fields import IntegerField
Expand Down Expand Up @@ -59,6 +61,7 @@
from databank.serializers import CountryOverviewSerializer
from deployments.models import ERU, Personnel
from deployments.serializers import ListDeployedERUByEventSerializer
from dref.models import Dref, DrefFinalReport, DrefOperationalUpdate
from main.enums import GlobalEnumSerializer, get_enum_values
from main.filters import NullsLastOrderingFilter
from main.permissions import DenyGuestUserMutationPermission, DenyGuestUserPermission
Expand All @@ -73,6 +76,7 @@
Appeal,
AppealDocument,
AppealHistory,
AppealStatus,
AppealType,
Country,
CountryKeyDocument,
Expand All @@ -86,6 +90,7 @@
EventContact,
EventFeaturedDocument,
EventSeverityLevelHistory,
EventStage,
Export,
ExternalPartner,
FieldReport,
Expand Down Expand Up @@ -749,7 +754,8 @@ def get_queryset(self, *args, **kwargs):
qset = super().get_queryset()
if self.action == "mini_events":
# return Event.objects.filter(parent_event__isnull=True).select_related('dtype')
return qset.filter(parent_event__isnull=True).select_related("dtype")
return qset.filter(parent_event__isnull=True).select_related("dtype").prefetch_related("countries_for_preview")

if self.action == "response_activity_events":
return (
qset.filter(parent_event__isnull=True)
Expand Down Expand Up @@ -869,7 +875,11 @@ def retrieve(self, request, pk=None, *args, **kwargs):
)
@action(methods=["get"], detail=False, url_path="mini")
def mini_events(self, request):
queryset = self.filter_queryset(self.get_queryset())
queryset = self.filter_queryset(self.get_queryset()).annotate(
latest_field_report_id=Subquery(
FieldReport.objects.filter(event=OuterRef("pk")).order_by("-updated_at").values("id")[:1]
)
)
serializer = ListMiniEventSerializer(queryset, many=True)
page = self.paginate_queryset(queryset)
if page is not None:
Expand Down Expand Up @@ -1559,45 +1569,181 @@ def get_queryset(self):
return CountrySupportingPartner.objects.select_related("country")


class EmergencyViewset(ReadOnlyVisibilityViewset):
class EmergencyViewset(
mixins.RetrieveModelMixin,
viewsets.GenericViewSet,
ReadOnlyVisibilityViewsetMixin,
):
queryset = Event.objects.all()
lookup_field = "id"
serializer_class = DetailEmergencySerializer
filterset_class = EventFilter
visibility_model_class = Event

def get_queryset(self):
today = timezone.now().date()

active_appeal_qs = Appeal.objects.filter(
event=OuterRef("pk"),
status=AppealStatus.ACTIVE,
)

active_dref_appeal_qs = active_appeal_qs.filter(
atype=AppealType.DREF,
)

active_emergency_appeal_qs = active_appeal_qs.filter(
atype=AppealType.APPEAL,
)

field_report_qs = FieldReport.objects.filter(event=OuterRef("pk"))

latest_appeal_qs = active_emergency_appeal_qs.order_by("-start_date")

approved_dref_qs = Dref.objects.filter(
event=OuterRef("pk"),
status=Dref.Status.APPROVED,
)

approved_ops_update_qs = DrefOperationalUpdate.objects.filter(
dref__event=OuterRef("pk"),
status=Dref.Status.APPROVED,
).order_by("-created_at")

approved_final_report_qs = DrefFinalReport.objects.filter(
dref__event=OuterRef("pk"),
status=Dref.Status.APPROVED,
).order_by("-created_at")

return (
super()
.get_queryset()
.select_related(
"dtype",
"parent_event",
)
.prefetch_related(
"regions",
"countries",
"countries_for_preview",
Prefetch("key_figures", queryset=KeyFigure.objects.all()),
Prefetch("contacts", queryset=EventContact.objects.all()),
.annotate(
# Aggregated Values
response_activity_count=Count(
"emergency_projects",
distinct=True,
),
active_deployments_count=Count(
"personneldeployment__personnel",
filter=Q(
personneldeployment__personnel__type=Personnel.TypeChoices.RR,
personneldeployment__personnel__start_date__date__lte=today,
personneldeployment__personnel__end_date__date__gte=today,
personneldeployment__personnel__is_active=True,
),
distinct=True,
),
surge_alerts_count=Count(
"surgealert",
distinct=True,
),
# Stage
stage=Case(
When(
Exists(active_emergency_appeal_qs),
then=Value(EventStage.EMERGENCY_APPEAL),
),
When(
Exists(approved_final_report_qs),
then=Value(EventStage.DREF_FINAL_REPORT),
),
When(
Exists(approved_ops_update_qs),
then=Value(EventStage.DREF_OPERATIONAL_UPDATE),
),
When(
Exists(approved_dref_qs),
then=Value(EventStage.DREF_APPLICATION),
),
# If there is an active appeal of DREF type, but no approved DREF yet,
# we consider the emergency to be in the Emergency Appeal stage.
# Reaches here only if no approved DREF/ops-update/final-report exists
# So an active appeal type DREF appeal with no approved DREF = Emergency Appeal
When(
Exists(active_dref_appeal_qs),
then=Value(EventStage.EMERGENCY_APPEAL),
),
When(
Exists(FieldReport.objects.filter(event=OuterRef("pk"))),
then=Value(EventStage.FIELD_REPORT),
),
default=Value(None),
output_field=IntegerField(null=True),
),
)
.annotate(
first_field_report_id=Subquery(
FieldReport.objects.filter(event=OuterRef("pk"))
.order_by(
"fr_num",
"updated_at",
)
.values("id")[:1]
# Passing values for the current stage's instance,
# to avoid extra queries in serializer.
stage_appeal_id=Case(
When(
stage=EventStage.EMERGENCY_APPEAL,
then=Subquery(latest_appeal_qs.values("id")[:1]),
),
default=Value(None),
output_field=IntegerField(null=True),
),
latest_field_report_id=Subquery(
FieldReport.objects.filter(event=OuterRef("pk"))
.order_by(
"-fr_num",
"-updated_at",
)
.values("id")[:1]
stage_dref_id=Case(
When(
stage__in=[
EventStage.DREF_APPLICATION,
EventStage.DREF_OPERATIONAL_UPDATE,
EventStage.DREF_FINAL_REPORT,
],
then=Subquery(approved_dref_qs.values("id")[:1]),
),
default=Value(None),
output_field=IntegerField(null=True),
),
stage_final_report_id=Case(
When(
stage=EventStage.DREF_FINAL_REPORT,
then=Subquery(
DrefFinalReport.objects.filter(
dref__id=OuterRef("stage_dref_id"), status=Dref.Status.APPROVED
).values("id")[:1]
),
),
default=Value(None),
output_field=IntegerField(null=True),
),
stage_ops_update_id=Case(
When(
stage=EventStage.DREF_OPERATIONAL_UPDATE,
then=Subquery(
DrefOperationalUpdate.objects.filter(
dref__id=OuterRef("stage_dref_id"), status=Dref.Status.APPROVED
).values("id")[:1]
),
),
default=Value(None),
output_field=IntegerField(null=True),
),
stage_field_report_id=Case(
When(
stage=EventStage.FIELD_REPORT,
then=Subquery(field_report_qs.order_by("-updated_at", "-fr_num").values("id")[:1]),
),
default=Value(None),
output_field=IntegerField(null=True),
),
first_field_report_created_at=Subquery(field_report_qs.order_by("created_at", "fr_num").values("created_at")[:1]),
latest_field_report_created_at=Subquery(
field_report_qs.order_by("-created_at", "-fr_num").values("created_at")[:1]
),
)
.select_related("dtype")
.prefetch_related(
Prefetch(
"countries",
queryset=Country.objects.select_related("region"),
),
Prefetch(
"key_figures",
queryset=KeyFigure.objects.all(),
),
Prefetch(
"contacts",
queryset=EventContact.objects.all(),
),
appeal_id=Subquery(Appeal.objects.filter(event=OuterRef("pk")).order_by("-created_at").values("id")[:1]),
)
)
1 change: 1 addition & 0 deletions api/factories/event.py
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ class Meta:
name = fuzzy.FuzzyText(length=50)
slug = fuzzy.FuzzyText(length=50)
dtype = factory.SubFactory(DisasterTypeFactory)
source = fuzzy.FuzzyChoice(Event.EventSource)

@factory.post_generation
def districts(self, create, extracted, **kwargs):
Expand Down
4 changes: 2 additions & 2 deletions api/management/commands/migrate_event_source.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,8 +19,8 @@ def handle(self, *args, **options):
event.source = Event.EventSource.GDACS
self.stdout.write(self.style.NOTICE(f"Updating {event.name} source to {Event.EventSource.GDACS.label}"))
else:
event.source = Event.EventSource.Manual_Input
self.stdout.write(self.style.NOTICE(f"Updating {event.name} source to {Event.EventSource.Manual_Input.label}"))
event.source = Event.EventSource.MANUAL_INPUT
self.stdout.write(self.style.NOTICE(f"Updating {event.name} source to {Event.EventSource.MANUAL_INPUT.label}"))
event.auto_generated = False
to_update.append(event)

Expand Down
14 changes: 12 additions & 2 deletions api/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -761,6 +761,15 @@ def snippet_image_path(instance, filename):
return "emergencies/%s/%s" % (instance.id, filename)


# NOTE: Stage for the emergency timeline
class EventStage(models.IntegerChoices):
EMERGENCY_APPEAL = 1, _("Emergency Appeal")
DREF_APPLICATION = 2, _("DREF Application")
DREF_OPERATIONAL_UPDATE = 3, _("DREF Operational Update")
DREF_FINAL_REPORT = 4, _("DREF Final Report")
FIELD_REPORT = 5, _("Field Report")


# NOTE: If ever in future we need to create an api to update the event table
# we also need to make sure to add appropriate signal to create ifrc severity level event history
@reversion.register()
Expand All @@ -769,7 +778,8 @@ class Event(models.Model):

class EventSource(models.IntegerChoices):

Manual_Input = 100, _("Manual input")
# TODO(Susilnem): use upper case
MANUAL_INPUT = 100, _("Manual input")
"""MANUAL_INPUT: Event data manually entered by a user through the event administration interface."""

GDACS = 110, _("GDACs scraper")
Expand Down Expand Up @@ -853,7 +863,7 @@ class EventSource(models.IntegerChoices):
previous_update = models.DateTimeField(verbose_name=_("previous update"), null=True, blank=True)

auto_generated = models.BooleanField(verbose_name=_("auto generated"), default=False, editable=False)
source = models.IntegerField(choices=EventSource.choices, default=EventSource.Manual_Input, verbose_name=_("Event source"))
source = models.IntegerField(choices=EventSource.choices, default=EventSource.MANUAL_INPUT, verbose_name=_("Event source"))

# Meant to give the organization a way of highlighting certain, important events.
is_featured = models.BooleanField(default=False, verbose_name=_("is featured on home page"))
Expand Down
Loading
Loading