Files
Backend/fertilization_recommendation/views.py
T

167 lines
5.9 KiB
Python
Raw Normal View History

"""
Fertilization Recommendation API views.
"""
2026-04-27 00:40:59 +03:30
import logging
2026-03-24 20:10:48 +03:30
from rest_framework import serializers, status
from rest_framework.response import Response
from rest_framework.views import APIView
2026-04-27 00:40:59 +03:30
from drf_spectacular.utils import extend_schema
2026-03-24 20:10:48 +03:30
from config.swagger import status_response
2026-03-25 00:51:04 +03:30
from external_api_adapter import request as external_api_request
2026-04-02 23:25:39 +03:30
from farm_hub.models import FarmHub
2026-03-25 00:51:04 +03:30
from .mock_data import CONFIG_RESPONSE_DATA
2026-04-02 23:25:39 +03:30
from .models import FertilizationRecommendationRequest
2026-03-26 15:39:31 +03:30
from .serializers import (
FertilizationRecommendRequestSerializer,
FertilizationRecommendResponseDataSerializer,
)
2026-04-27 00:40:59 +03:30
logger = logging.getLogger(__name__)
2026-04-02 23:25:39 +03:30
class FarmAccessMixin:
@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
class ConfigView(FarmAccessMixin, APIView):
2026-03-24 20:10:48 +03:30
@extend_schema(
tags=["Fertilization Recommendation"],
responses={200: status_response("FertilizationConfigResponse", data=serializers.JSONField())},
)
def get(self, request):
2026-04-02 23:25:39 +03:30
farm = self._get_farm(request, request.query_params.get("farm_uuid"))
data = dict(CONFIG_RESPONSE_DATA)
data["farm_uuid"] = str(farm.farm_uuid)
return Response({"status": "success", "data": data}, status=status.HTTP_200_OK)
2026-04-02 23:25:39 +03:30
class RecommendView(FarmAccessMixin, APIView):
2026-04-27 00:40:59 +03:30
@staticmethod
def _normalize_sections(raw_sections):
if not isinstance(raw_sections, list):
return []
allowed_keys = {
"type",
"title",
"icon",
"content",
"items",
"fertilizerType",
"amount",
"applicationMethod",
"timing",
"validityPeriod",
"expandableExplanation",
}
normalized_sections = []
for section in raw_sections:
if not isinstance(section, dict) or not section.get("type"):
continue
normalized_section = {}
for key in allowed_keys:
value = section.get(key)
if value is None:
continue
if key == "items":
if not isinstance(value, list):
continue
normalized_section[key] = [str(item) for item in value]
continue
normalized_section[key] = str(value) if key != "type" else value
normalized_sections.append(normalized_section)
return normalized_sections
def _extract_public_sections(self, adapter_data):
if not isinstance(adapter_data, dict):
return []
data = adapter_data.get("data")
if isinstance(data, dict) and isinstance(data.get("sections"), list):
return self._normalize_sections(data.get("sections"))
result = data.get("result") if isinstance(data, dict) else None
if isinstance(result, dict) and isinstance(result.get("sections"), list):
return self._normalize_sections(result.get("sections"))
if isinstance(adapter_data.get("sections"), list):
return self._normalize_sections(adapter_data.get("sections"))
return []
2026-03-24 20:10:48 +03:30
@extend_schema(
tags=["Fertilization Recommendation"],
2026-03-26 15:39:31 +03:30
request=FertilizationRecommendRequestSerializer,
responses={200: status_response("FertilizationRecommendResponse", data=FertilizationRecommendResponseDataSerializer())},
2026-03-24 20:10:48 +03:30
)
def post(self, request):
2026-04-02 23:25:39 +03:30
serializer = FertilizationRecommendRequestSerializer(data=request.data)
serializer.is_valid(raise_exception=True)
payload = serializer.validated_data.copy()
farm = self._get_farm(request, payload.get("farm_uuid"))
payload["farm_uuid"] = str(farm.farm_uuid)
2026-04-27 00:40:59 +03:30
payload["plant_name"] = payload.get("plant_name", "")
payload["growth_stage"] = payload.get("growth_stage", "")
2026-04-02 23:25:39 +03:30
2026-03-25 00:51:04 +03:30
adapter_response = external_api_request(
"ai",
2026-04-27 00:40:59 +03:30
"/api/fertilization/recommend/",
2026-03-25 00:51:04 +03:30
method="POST",
2026-04-02 23:25:39 +03:30
payload=payload,
)
2026-03-26 15:39:31 +03:30
2026-04-02 23:25:39 +03:30
response_data = adapter_response.data if isinstance(adapter_response.data, dict) else {}
2026-04-27 00:40:59 +03:30
public_sections = self._extract_public_sections(response_data)
logger.warning(
"Fertilization recommendation response parsed: farm_uuid=%s status_code=%s response_keys=%s sections_count=%s",
str(farm.farm_uuid),
adapter_response.status_code,
sorted(response_data.keys()) if isinstance(response_data, dict) else None,
len(public_sections),
)
2026-04-02 23:25:39 +03:30
FertilizationRecommendationRequest.objects.create(
farm=farm,
2026-04-27 00:40:59 +03:30
crop_id=payload.get("plant_name", ""),
2026-04-02 23:25:39 +03:30
growth_stage=payload.get("growth_stage", ""),
2026-04-27 00:40:59 +03:30
task_id="",
status="success" if adapter_response.status_code < 400 else "error",
2026-04-02 23:25:39 +03:30
request_payload=payload,
response_payload=adapter_response.data if isinstance(adapter_response.data, dict) else {"raw": adapter_response.data},
)
2026-04-27 00:40:59 +03:30
if adapter_response.status_code >= 400:
return Response(
{
"code": adapter_response.status_code,
"msg": "error",
"data": response_data if isinstance(response_data, dict) else {"message": str(adapter_response.data)},
},
status=adapter_response.status_code,
)
2026-03-26 15:39:31 +03:30
2026-04-27 00:40:59 +03:30
return Response(
{
"code": 200,
"msg": "success",
"data": {
"sections": public_sections,
},
},
status=status.HTTP_200_OK,
2026-03-26 15:39:31 +03:30
)