2026-04-29 03:47:48 +03:30
|
|
|
import logging
|
|
|
|
|
|
2026-04-29 02:58:56 +03:30
|
|
|
from drf_spectacular.utils import OpenApiExample, extend_schema
|
|
|
|
|
from rest_framework import serializers, status
|
|
|
|
|
from rest_framework.permissions import IsAuthenticated
|
2026-04-10 16:12:51 +03:30
|
|
|
from rest_framework.response import Response
|
|
|
|
|
from rest_framework.views import APIView
|
|
|
|
|
|
2026-05-05 21:01:58 +03:30
|
|
|
from config.integration_contract import build_integration_meta
|
2026-04-29 02:58:56 +03:30
|
|
|
from config.swagger import code_response
|
|
|
|
|
from farm_hub.models import FarmHub
|
2026-04-10 16:12:51 +03:30
|
|
|
|
2026-04-29 02:58:56 +03:30
|
|
|
from .serializers import AlertTrackerAIResponseSerializer, FarmAlertsTrackerRequestSerializer
|
2026-04-29 03:47:48 +03:30
|
|
|
from .services import AlertService, build_tracker_response_from_snapshot
|
|
|
|
|
|
|
|
|
|
logger = logging.getLogger("farm_alerts")
|
2026-04-10 16:12:51 +03:30
|
|
|
|
|
|
|
|
|
2026-04-27 00:40:59 +03:30
|
|
|
class FarmAlertsBaseView(APIView):
|
2026-04-29 02:58:56 +03:30
|
|
|
permission_classes = [IsAuthenticated]
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
|
def _get_farm(request, farm_uuid):
|
|
|
|
|
if not farm_uuid:
|
|
|
|
|
raise serializers.ValidationError({"farm_uuid": ["This field is required."]})
|
|
|
|
|
try:
|
|
|
|
|
return FarmHub.objects.get(farm_uuid=farm_uuid, owner=request.user)
|
|
|
|
|
except FarmHub.DoesNotExist as exc:
|
|
|
|
|
raise serializers.ValidationError({"farm_uuid": ["Farm not found."]}) from exc
|
|
|
|
|
|
2026-04-10 16:12:51 +03:30
|
|
|
|
2026-04-27 00:40:59 +03:30
|
|
|
class AlertTrackerView(FarmAlertsBaseView):
|
2026-04-29 02:58:56 +03:30
|
|
|
@extend_schema(
|
|
|
|
|
tags=["Farm Alerts"],
|
|
|
|
|
request=FarmAlertsTrackerRequestSerializer,
|
|
|
|
|
examples=[
|
|
|
|
|
OpenApiExample(
|
|
|
|
|
"Tracker Request",
|
|
|
|
|
value={
|
|
|
|
|
"farm_uuid": "11111111-1111-1111-1111-111111111111",
|
|
|
|
|
},
|
|
|
|
|
request_only=True,
|
|
|
|
|
)
|
|
|
|
|
],
|
|
|
|
|
responses={200: code_response("FarmAlertsTrackerResponse", data=AlertTrackerAIResponseSerializer())},
|
|
|
|
|
)
|
2026-04-10 16:12:51 +03:30
|
|
|
def post(self, request):
|
2026-04-29 02:58:56 +03:30
|
|
|
request_serializer = FarmAlertsTrackerRequestSerializer(data=request.data)
|
|
|
|
|
request_serializer.is_valid(raise_exception=True)
|
2026-04-10 16:12:51 +03:30
|
|
|
|
2026-04-29 02:58:56 +03:30
|
|
|
farm = self._get_farm(request, request_serializer.validated_data["farm_uuid"])
|
2026-04-29 03:47:48 +03:30
|
|
|
logger.info(
|
|
|
|
|
"tracker endpoint received request farm=%s payload=%s",
|
|
|
|
|
farm.farm_uuid,
|
|
|
|
|
request.data,
|
2026-04-10 16:12:51 +03:30
|
|
|
)
|
|
|
|
|
|
2026-04-29 03:47:48 +03:30
|
|
|
response_data = build_tracker_response_from_snapshot(farm=farm)
|
|
|
|
|
logger.info(
|
|
|
|
|
"tracker endpoint returning cached response farm=%s response=%s",
|
|
|
|
|
farm.farm_uuid,
|
|
|
|
|
response_data,
|
|
|
|
|
)
|
2026-04-29 02:58:56 +03:30
|
|
|
serializer = AlertTrackerAIResponseSerializer(instance=response_data)
|
2026-05-05 21:01:58 +03:30
|
|
|
snapshot = getattr(farm, "alert_tracker_snapshot", None)
|
|
|
|
|
return Response(
|
|
|
|
|
{
|
|
|
|
|
"code": 200,
|
|
|
|
|
"msg": "success",
|
|
|
|
|
"data": serializer.data,
|
|
|
|
|
"meta": build_integration_meta(
|
|
|
|
|
flow_type="cached_snapshot",
|
|
|
|
|
source_type="cached_snapshot",
|
|
|
|
|
source_service="backend_farm_alerts_snapshot",
|
|
|
|
|
ownership="backend",
|
|
|
|
|
live=False,
|
|
|
|
|
cached=True,
|
|
|
|
|
snapshot_at=getattr(snapshot, "updated_at", None),
|
|
|
|
|
notes=["Returns persisted tracker snapshot, not live AI inference."],
|
|
|
|
|
),
|
|
|
|
|
},
|
|
|
|
|
status=status.HTTP_200_OK,
|
|
|
|
|
)
|