2026-04-27 00:40:59 +03:30
|
|
|
from unittest.mock import patch
|
|
|
|
|
|
2026-04-30 01:01:04 +03:30
|
|
|
from django.core.cache import cache
|
2026-04-27 00:40:59 +03:30
|
|
|
from django.contrib.auth import get_user_model
|
2026-04-30 01:01:04 +03:30
|
|
|
from django.test import TestCase, override_settings
|
2026-04-27 00:40:59 +03:30
|
|
|
from django.urls import Resolver404, resolve
|
|
|
|
|
from rest_framework.test import APIRequestFactory, force_authenticate
|
|
|
|
|
|
|
|
|
|
from external_api_adapter.adapter import AdapterResponse
|
|
|
|
|
from farm_hub.models import FarmHub, FarmType
|
|
|
|
|
|
2026-04-30 01:01:04 +03:30
|
|
|
from .views import WaterNeedPredictionView, WaterSummaryView, WeatherFarmCardView
|
2026-04-27 00:40:59 +03:30
|
|
|
|
|
|
|
|
|
2026-04-30 01:01:04 +03:30
|
|
|
TEST_CACHES = {
|
|
|
|
|
"default": {
|
|
|
|
|
"BACKEND": "django.core.cache.backends.locmem.LocMemCache",
|
|
|
|
|
"LOCATION": "water-tests",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_WATER_NEED_PREDICTION_CACHE_TTL = 14400
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@override_settings(
|
|
|
|
|
CACHES=TEST_CACHES,
|
|
|
|
|
WATER_NEED_PREDICTION_CACHE_TTL=TEST_WATER_NEED_PREDICTION_CACHE_TTL,
|
|
|
|
|
)
|
2026-04-27 00:40:59 +03:30
|
|
|
class WeatherViewTests(TestCase):
|
|
|
|
|
def setUp(self):
|
2026-04-30 01:01:04 +03:30
|
|
|
cache.clear()
|
2026-04-27 00:40:59 +03:30
|
|
|
self.factory = APIRequestFactory()
|
|
|
|
|
self.user = get_user_model().objects.create_user(
|
|
|
|
|
username="farmer",
|
|
|
|
|
password="secret123",
|
|
|
|
|
email="farmer@example.com",
|
|
|
|
|
phone_number="09120000000",
|
|
|
|
|
)
|
|
|
|
|
self.other_user = get_user_model().objects.create_user(
|
|
|
|
|
username="other-farmer",
|
|
|
|
|
password="secret123",
|
|
|
|
|
email="other@example.com",
|
|
|
|
|
phone_number="09120000001",
|
|
|
|
|
)
|
|
|
|
|
self.farm_type = FarmType.objects.create(name="زراعی")
|
|
|
|
|
self.farm = FarmHub.objects.create(owner=self.user, farm_type=self.farm_type, name="Farm 1")
|
|
|
|
|
self.other_farm = FarmHub.objects.create(owner=self.other_user, farm_type=self.farm_type, name="Farm 2")
|
|
|
|
|
|
|
|
|
|
@patch("water.views.external_api_request")
|
|
|
|
|
def test_farm_card_proxies_to_ai_service(self, mock_external_api_request):
|
|
|
|
|
mock_external_api_request.return_value = AdapterResponse(
|
|
|
|
|
status_code=200,
|
|
|
|
|
data={"data": {"result": {"condition": "صاف", "temperature": 28.0}}},
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
request = self.factory.post("/api/weather/farm-card/", {"farm_uuid": str(self.farm.farm_uuid)}, format="json")
|
|
|
|
|
force_authenticate(request, user=self.user)
|
|
|
|
|
|
|
|
|
|
response = WeatherFarmCardView.as_view()(request)
|
|
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
self.assertEqual(response.data["code"], 200)
|
|
|
|
|
self.assertEqual(response.data["data"]["condition"], "صاف")
|
|
|
|
|
mock_external_api_request.assert_called_once_with(
|
|
|
|
|
"ai",
|
|
|
|
|
"/api/weather/farm-card/",
|
|
|
|
|
method="POST",
|
|
|
|
|
payload={"farm_uuid": str(self.farm.farm_uuid)},
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
@patch("water.views.external_api_request")
|
|
|
|
|
def test_get_water_need_prediction_uses_same_ai_service_for_farm_uuid(self, mock_external_api_request):
|
|
|
|
|
mock_external_api_request.return_value = AdapterResponse(
|
|
|
|
|
status_code=200,
|
|
|
|
|
data={"data": {"result": {"totalNext7Days": 24.6, "unit": "mm"}}},
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
request = self.factory.get(f"/api/water/need-prediction/?farm_uuid={self.farm.farm_uuid}")
|
|
|
|
|
|
|
|
|
|
response = WaterNeedPredictionView.as_view()(request)
|
|
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
self.assertEqual(response.data["data"]["farm_uuid"], str(self.farm.farm_uuid))
|
|
|
|
|
self.assertEqual(response.data["data"]["totalNext7Days"], 24.6)
|
|
|
|
|
mock_external_api_request.assert_called_once_with(
|
|
|
|
|
"ai",
|
|
|
|
|
"/api/weather/water-need-prediction/",
|
|
|
|
|
method="POST",
|
|
|
|
|
payload={"farm_uuid": str(self.farm.farm_uuid)},
|
|
|
|
|
)
|
|
|
|
|
|
2026-04-30 01:01:04 +03:30
|
|
|
@patch("water.views.external_api_request")
|
|
|
|
|
def test_water_need_prediction_caches_last_four_ai_responses(self, mock_external_api_request):
|
|
|
|
|
farms = []
|
|
|
|
|
for index in range(5):
|
|
|
|
|
farm = FarmHub.objects.create(owner=self.user, farm_type=self.farm_type, name=f"Farm Cache {index}")
|
|
|
|
|
farms.append(farm)
|
|
|
|
|
|
|
|
|
|
mock_external_api_request.return_value = AdapterResponse(
|
|
|
|
|
status_code=200,
|
|
|
|
|
data={"data": {"result": {"totalNext7Days": float(index), "unit": "mm"}}},
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
request = self.factory.get(f"/api/water/need-prediction/?farm_uuid={farm.farm_uuid}")
|
|
|
|
|
response = WaterNeedPredictionView.as_view()(request)
|
|
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
|
|
cached_items = cache.get(WeatherFarmCardView.WATER_NEED_PREDICTION_CACHE_KEY)
|
|
|
|
|
|
|
|
|
|
self.assertEqual(len(cached_items), 4)
|
|
|
|
|
self.assertEqual(cached_items[0]["totalNext7Days"], 4.0)
|
|
|
|
|
self.assertEqual(cached_items[-1]["totalNext7Days"], 1.0)
|
|
|
|
|
|
|
|
|
|
@patch("water.views.external_api_request")
|
|
|
|
|
def test_water_need_prediction_returns_cached_response_for_same_farm(self, mock_external_api_request):
|
|
|
|
|
mock_external_api_request.return_value = AdapterResponse(
|
|
|
|
|
status_code=200,
|
|
|
|
|
data={"data": {"result": {"totalNext7Days": 24.6, "unit": "mm"}}},
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
for _ in range(2):
|
|
|
|
|
request = self.factory.get(f"/api/water/need-prediction/?farm_uuid={self.farm.farm_uuid}")
|
|
|
|
|
force_authenticate(request, user=self.user)
|
|
|
|
|
response = WaterNeedPredictionView.as_view()(request)
|
|
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
self.assertEqual(response.data["data"]["farm_uuid"], str(self.farm.farm_uuid))
|
|
|
|
|
|
|
|
|
|
cache_key = WeatherFarmCardView._build_water_need_prediction_cache_key(self.user.id, self.farm.farm_uuid)
|
|
|
|
|
self.assertEqual(cache.get(cache_key)["totalNext7Days"], 24.6)
|
|
|
|
|
mock_external_api_request.assert_called_once()
|
|
|
|
|
|
|
|
|
|
@patch("water.views.cache.set")
|
|
|
|
|
@patch("water.views.external_api_request")
|
|
|
|
|
def test_water_need_prediction_uses_env_ttl_for_cache(self, mock_external_api_request, mock_cache_set):
|
|
|
|
|
mock_external_api_request.return_value = AdapterResponse(
|
|
|
|
|
status_code=200,
|
|
|
|
|
data={"data": {"result": {"totalNext7Days": 24.6, "unit": "mm"}}},
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
request = self.factory.get(f"/api/water/need-prediction/?farm_uuid={self.farm.farm_uuid}")
|
|
|
|
|
force_authenticate(request, user=self.user)
|
|
|
|
|
response = WaterNeedPredictionView.as_view()(request)
|
|
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
self.assertTrue(
|
|
|
|
|
any(call.kwargs.get("timeout") == TEST_WATER_NEED_PREDICTION_CACHE_TTL for call in mock_cache_set.call_args_list)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
def test_water_summary_caches_last_four_responses(self):
|
|
|
|
|
for index in range(5):
|
|
|
|
|
farm = FarmHub.objects.create(owner=self.user, farm_type=self.farm_type, name=f"Summary Farm {index}")
|
|
|
|
|
request = self.factory.get(f"/api/water/summary/?farm_uuid={farm.farm_uuid}")
|
|
|
|
|
response = WaterSummaryView.as_view()(request)
|
|
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
cached_items = cache.get(WeatherFarmCardView.WATER_SUMMARY_CACHE_KEY)
|
|
|
|
|
cached_items[0]["farmWeatherCard"]["condition"] = f"Condition {index}"
|
|
|
|
|
cache.set(WeatherFarmCardView.WATER_SUMMARY_CACHE_KEY, cached_items, timeout=None)
|
|
|
|
|
|
|
|
|
|
cached_items = cache.get(WeatherFarmCardView.WATER_SUMMARY_CACHE_KEY)
|
|
|
|
|
|
|
|
|
|
self.assertEqual(len(cached_items), 4)
|
|
|
|
|
self.assertEqual(cached_items[0]["farmWeatherCard"]["condition"], "Condition 4")
|
|
|
|
|
self.assertEqual(cached_items[-1]["farmWeatherCard"]["condition"], "Condition 1")
|
|
|
|
|
|
2026-04-27 00:40:59 +03:30
|
|
|
def test_weather_view_rejects_foreign_farm_uuid(self):
|
|
|
|
|
request = self.factory.post("/api/weather/farm-card/", {"farm_uuid": str(self.other_farm.farm_uuid)}, format="json")
|
|
|
|
|
force_authenticate(request, user=self.user)
|
|
|
|
|
|
|
|
|
|
response = WeatherFarmCardView.as_view()(request)
|
|
|
|
|
|
|
|
|
|
self.assertEqual(response.status_code, 404)
|
|
|
|
|
self.assertEqual(response.data["code"], 404)
|
|
|
|
|
|
|
|
|
|
def test_weather_post_routes_exist_only_under_weather_prefix(self):
|
|
|
|
|
self.assertIs(resolve("/api/weather/farm-card/").func.view_class, WeatherFarmCardView)
|
|
|
|
|
|
|
|
|
|
with self.assertRaises(Resolver404):
|
|
|
|
|
resolve("/api/water/farm-card/")
|
|
|
|
|
|
|
|
|
|
with self.assertRaises(Resolver404):
|
|
|
|
|
resolve("/api/water/water-need-prediction/")
|
|
|
|
|
|
|
|
|
|
def test_water_get_routes_do_not_exist_under_weather_prefix(self):
|
|
|
|
|
with self.assertRaises(Resolver404):
|
|
|
|
|
resolve("/api/weather/card/")
|
|
|
|
|
|
|
|
|
|
with self.assertRaises(Resolver404):
|
|
|
|
|
resolve("/api/weather/need-prediction/")
|
|
|
|
|
|
|
|
|
|
with self.assertRaises(Resolver404):
|
|
|
|
|
resolve("/api/weather/water-need-prediction/")
|
|
|
|
|
|
|
|
|
|
with self.assertRaises(Resolver404):
|
|
|
|
|
resolve("/api/weather/stress-index/")
|
|
|
|
|
|
|
|
|
|
with self.assertRaises(Resolver404):
|
|
|
|
|
resolve("/api/weather/summary/")
|