first commit
This commit is contained in:
@@ -0,0 +1 @@
|
||||
|
||||
@@ -0,0 +1,19 @@
|
||||
from django.contrib import admin
|
||||
|
||||
from .models import Plant
|
||||
|
||||
|
||||
@admin.register(Plant)
|
||||
class PlantAdmin(admin.ModelAdmin):
|
||||
list_display = (
|
||||
"id",
|
||||
"name",
|
||||
"light",
|
||||
"soil",
|
||||
"temperature",
|
||||
"planting_season",
|
||||
"created_at",
|
||||
)
|
||||
list_filter = ("planting_season",)
|
||||
search_fields = ("name",)
|
||||
readonly_fields = ("created_at", "updated_at")
|
||||
@@ -0,0 +1,7 @@
|
||||
from django.apps import AppConfig
|
||||
|
||||
|
||||
class PlantConfig(AppConfig):
|
||||
default_auto_field = "django.db.models.BigAutoField"
|
||||
name = "plant"
|
||||
verbose_name = "Plant"
|
||||
@@ -0,0 +1 @@
|
||||
|
||||
@@ -0,0 +1 @@
|
||||
|
||||
@@ -0,0 +1,109 @@
|
||||
"""
|
||||
Management command to seed initial plant data.
|
||||
Run: python manage.py seed_plants
|
||||
"""
|
||||
|
||||
from django.core.management.base import BaseCommand
|
||||
|
||||
from plant.models import Plant
|
||||
|
||||
|
||||
INITIAL_PLANTS = [
|
||||
{
|
||||
"name": "گوجهفرنگی",
|
||||
"light": "آفتاب کامل (۶-۸ ساعت)",
|
||||
"watering": "منظم، هفتهای ۲-۳ بار",
|
||||
"soil": "لومی، غنی از مواد آلی، pH بین ۶-۶.۸",
|
||||
"temperature": "۲۰-۳۰ درجه سانتیگراد",
|
||||
"planting_season": "بهار",
|
||||
"harvest_time": "۷۰-۹۰ روز پس از کاشت",
|
||||
"spacing": "۴۵-۶۰ سانتیمتر",
|
||||
"fertilizer": "کود NPK متعادل، کمپوست",
|
||||
},
|
||||
{
|
||||
"name": "خیار",
|
||||
"light": "آفتاب کامل",
|
||||
"watering": "روزانه در فصل گرم",
|
||||
"soil": "لومی شنی، غنی از هوموس",
|
||||
"temperature": "۱۸-۳۰ درجه سانتیگراد",
|
||||
"planting_season": "بهار تا اوایل تابستان",
|
||||
"harvest_time": "۵۰-۷۰ روز پس از کاشت",
|
||||
"spacing": "۳۰-۴۵ سانتیمتر",
|
||||
"fertilizer": "کود ازته، کمپوست",
|
||||
},
|
||||
{
|
||||
"name": "فلفل دلمهای",
|
||||
"light": "آفتاب کامل (۶-۸ ساعت)",
|
||||
"watering": "منظم، هفتهای ۲-۳ بار",
|
||||
"soil": "لومی، زهکشی مناسب",
|
||||
"temperature": "۲۰-۳۰ درجه سانتیگراد",
|
||||
"planting_season": "بهار",
|
||||
"harvest_time": "۶۰-۹۰ روز پس از کاشت",
|
||||
"spacing": "۴۰-۵۰ سانتیمتر",
|
||||
"fertilizer": "کود فسفره و پتاسه",
|
||||
},
|
||||
{
|
||||
"name": "هویج",
|
||||
"light": "آفتاب کامل تا نیمهسایه",
|
||||
"watering": "منظم، خاک مرطوب",
|
||||
"soil": "شنی لومی، عمیق، بدون سنگ",
|
||||
"temperature": "۱۵-۲۵ درجه سانتیگراد",
|
||||
"planting_season": "اوایل بهار یا پاییز",
|
||||
"harvest_time": "۷۰-۸۰ روز پس از کاشت",
|
||||
"spacing": "۵-۸ سانتیمتر",
|
||||
"fertilizer": "کود پتاسه، کمپوست پوسیده",
|
||||
},
|
||||
{
|
||||
"name": "کاهو",
|
||||
"light": "نیمهسایه تا آفتاب کامل",
|
||||
"watering": "منظم، خاک مرطوب",
|
||||
"soil": "لومی، غنی از مواد آلی",
|
||||
"temperature": "۱۰-۲۰ درجه سانتیگراد",
|
||||
"planting_season": "بهار و پاییز",
|
||||
"harvest_time": "۴۵-۶۰ روز پس از کاشت",
|
||||
"spacing": "۲۰-۳۰ سانتیمتر",
|
||||
"fertilizer": "کود ازته، کمپوست",
|
||||
},
|
||||
{
|
||||
"name": "سیبزمینی",
|
||||
"light": "آفتاب کامل",
|
||||
"watering": "منظم، هفتهای ۲ بار",
|
||||
"soil": "لومی شنی، اسیدی ملایم، pH بین ۵-۶",
|
||||
"temperature": "۱۵-۲۲ درجه سانتیگراد",
|
||||
"planting_season": "اواخر زمستان تا اوایل بهار",
|
||||
"harvest_time": "۹۰-۱۲۰ روز پس از کاشت",
|
||||
"spacing": "۳۰-۴۰ سانتیمتر",
|
||||
"fertilizer": "کود NPK، کمپوست",
|
||||
},
|
||||
{
|
||||
"name": "پیاز",
|
||||
"light": "آفتاب کامل",
|
||||
"watering": "منظم، خاک مرطوب ولی نه غرقابی",
|
||||
"soil": "لومی، زهکشی خوب",
|
||||
"temperature": "۱۲-۲۴ درجه سانتیگراد",
|
||||
"planting_season": "پاییز یا اوایل بهار",
|
||||
"harvest_time": "۹۰-۱۵۰ روز پس از کاشت",
|
||||
"spacing": "۱۰-۱۵ سانتیمتر",
|
||||
"fertilizer": "کود فسفره، سولفات پتاسیم",
|
||||
},
|
||||
]
|
||||
|
||||
|
||||
class Command(BaseCommand):
|
||||
help = "Seed initial plant data (7 common vegetables)"
|
||||
|
||||
def handle(self, *args, **options):
|
||||
created_count = 0
|
||||
for plant_data in INITIAL_PLANTS:
|
||||
_, created = Plant.objects.get_or_create(
|
||||
name=plant_data["name"],
|
||||
defaults=plant_data,
|
||||
)
|
||||
if created:
|
||||
created_count += 1
|
||||
self.stdout.write(
|
||||
self.style.SUCCESS(f" Created: {plant_data['name']}")
|
||||
)
|
||||
self.stdout.write(
|
||||
self.style.SUCCESS(f"\nDone. Created {created_count} new plants.")
|
||||
)
|
||||
@@ -0,0 +1,36 @@
|
||||
# Generated by Django 5.2.12 on 2026-03-19 15:01
|
||||
|
||||
from django.db import migrations, models
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
initial = True
|
||||
|
||||
dependencies = [
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.CreateModel(
|
||||
name='Plant',
|
||||
fields=[
|
||||
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('name', models.CharField(db_index=True, help_text='نام گیاه', max_length=255, unique=True)),
|
||||
('light', models.CharField(blank=True, help_text='نور مورد نیاز', max_length=255)),
|
||||
('watering', models.CharField(blank=True, help_text='آبیاری', max_length=255)),
|
||||
('soil', models.CharField(blank=True, help_text='خاک مناسب', max_length=255)),
|
||||
('temperature', models.CharField(blank=True, help_text='دمای مناسب', max_length=255)),
|
||||
('planting_season', models.CharField(blank=True, help_text='فصل کاشت', max_length=255)),
|
||||
('harvest_time', models.CharField(blank=True, help_text='زمان برداشت', max_length=255)),
|
||||
('spacing', models.CharField(blank=True, help_text='فاصله کاشت', max_length=255)),
|
||||
('fertilizer', models.CharField(blank=True, help_text='کود مناسب', max_length=255)),
|
||||
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||
('updated_at', models.DateTimeField(auto_now=True)),
|
||||
],
|
||||
options={
|
||||
'verbose_name': 'گیاه',
|
||||
'verbose_name_plural': 'گیاهان',
|
||||
'ordering': ['name'],
|
||||
},
|
||||
),
|
||||
]
|
||||
@@ -0,0 +1 @@
|
||||
|
||||
@@ -0,0 +1,64 @@
|
||||
from django.db import models
|
||||
|
||||
|
||||
class Plant(models.Model):
|
||||
"""
|
||||
اطلاعات گیاهان شامل شرایط نگهداری و کاشت.
|
||||
"""
|
||||
|
||||
name = models.CharField(
|
||||
max_length=255,
|
||||
unique=True,
|
||||
db_index=True,
|
||||
help_text="نام گیاه",
|
||||
)
|
||||
light = models.CharField(
|
||||
max_length=255,
|
||||
blank=True,
|
||||
help_text="نور مورد نیاز",
|
||||
)
|
||||
watering = models.CharField(
|
||||
max_length=255,
|
||||
blank=True,
|
||||
help_text="آبیاری",
|
||||
)
|
||||
soil = models.CharField(
|
||||
max_length=255,
|
||||
blank=True,
|
||||
help_text="خاک مناسب",
|
||||
)
|
||||
temperature = models.CharField(
|
||||
max_length=255,
|
||||
blank=True,
|
||||
help_text="دمای مناسب",
|
||||
)
|
||||
planting_season = models.CharField(
|
||||
max_length=255,
|
||||
blank=True,
|
||||
help_text="فصل کاشت",
|
||||
)
|
||||
harvest_time = models.CharField(
|
||||
max_length=255,
|
||||
blank=True,
|
||||
help_text="زمان برداشت",
|
||||
)
|
||||
spacing = models.CharField(
|
||||
max_length=255,
|
||||
blank=True,
|
||||
help_text="فاصله کاشت",
|
||||
)
|
||||
fertilizer = models.CharField(
|
||||
max_length=255,
|
||||
blank=True,
|
||||
help_text="کود مناسب",
|
||||
)
|
||||
created_at = models.DateTimeField(auto_now_add=True)
|
||||
updated_at = models.DateTimeField(auto_now=True)
|
||||
|
||||
class Meta:
|
||||
ordering = ["name"]
|
||||
verbose_name = "گیاه"
|
||||
verbose_name_plural = "گیاهان"
|
||||
|
||||
def __str__(self):
|
||||
return self.name
|
||||
@@ -0,0 +1,25 @@
|
||||
from rest_framework import serializers
|
||||
|
||||
from .models import Plant
|
||||
|
||||
|
||||
class PlantSerializer(serializers.ModelSerializer):
|
||||
"""سریالایزر خروجی / ورودی برای Plant."""
|
||||
|
||||
class Meta:
|
||||
model = Plant
|
||||
fields = [
|
||||
"id",
|
||||
"name",
|
||||
"light",
|
||||
"watering",
|
||||
"soil",
|
||||
"temperature",
|
||||
"planting_season",
|
||||
"harvest_time",
|
||||
"spacing",
|
||||
"fertilizer",
|
||||
"created_at",
|
||||
"updated_at",
|
||||
]
|
||||
read_only_fields = ["id", "created_at", "updated_at"]
|
||||
@@ -0,0 +1,34 @@
|
||||
"""
|
||||
سرویسهای گیاه — دریافت مشخصات گیاه از API خارجی بر اساس نام.
|
||||
"""
|
||||
|
||||
import logging
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
|
||||
def fetch_plant_info_from_api(plant_name: str) -> dict | None:
|
||||
"""
|
||||
اتصال به API خارجی و دریافت مشخصات گیاه بر اساس نام.
|
||||
|
||||
TODO: پیادهسازی اتصال واقعی به API.
|
||||
در حال حاضر این تابع خالی است و None برمیگرداند.
|
||||
|
||||
پارامترها:
|
||||
plant_name: نام گیاه
|
||||
|
||||
خروجی مورد انتظار (وقتی پیادهسازی شود):
|
||||
{
|
||||
"name": "گوجهفرنگی",
|
||||
"light": "آفتاب کامل",
|
||||
"watering": "منظم، هفتهای ۲-۳ بار",
|
||||
"soil": "لومی، غنی از مواد آلی",
|
||||
"temperature": "۲۰-۳۰ درجه سانتیگراد",
|
||||
"planting_season": "بهار",
|
||||
"harvest_time": "۷۰-۹۰ روز پس از کاشت",
|
||||
"spacing": "۴۵-۶۰ سانتیمتر",
|
||||
"fertilizer": "کود NPK متعادل",
|
||||
}
|
||||
"""
|
||||
# TODO: اتصال واقعی به API
|
||||
return None
|
||||
@@ -0,0 +1,9 @@
|
||||
from django.urls import path
|
||||
|
||||
from .views import PlantDetailView, PlantFetchInfoView, PlantListCreateView
|
||||
|
||||
urlpatterns = [
|
||||
path("", PlantListCreateView.as_view(), name="plant-list-create"),
|
||||
path("<int:pk>/", PlantDetailView.as_view(), name="plant-detail"),
|
||||
path("fetch-info/", PlantFetchInfoView.as_view(), name="plant-fetch-info"),
|
||||
]
|
||||
+234
@@ -0,0 +1,234 @@
|
||||
from drf_spectacular.utils import (
|
||||
OpenApiExample,
|
||||
OpenApiResponse,
|
||||
extend_schema,
|
||||
inline_serializer,
|
||||
)
|
||||
from rest_framework import serializers as drf_serializers
|
||||
from rest_framework import status
|
||||
from rest_framework.response import Response
|
||||
from rest_framework.views import APIView
|
||||
|
||||
from .models import Plant
|
||||
from .serializers import PlantSerializer
|
||||
from .services import fetch_plant_info_from_api
|
||||
|
||||
|
||||
class PlantListCreateView(APIView):
|
||||
"""لیست تمام گیاهان و ایجاد گیاه جدید."""
|
||||
|
||||
@extend_schema(
|
||||
tags=["Plant"],
|
||||
summary="لیست گیاهان",
|
||||
description="لیست تمام گیاهان ذخیرهشده را برمیگرداند.",
|
||||
responses={200: PlantSerializer(many=True)},
|
||||
)
|
||||
def get(self, request):
|
||||
plants = Plant.objects.all()
|
||||
serializer = PlantSerializer(plants, many=True)
|
||||
return Response(
|
||||
{"code": 200, "msg": "success", "data": serializer.data},
|
||||
status=status.HTTP_200_OK,
|
||||
)
|
||||
|
||||
@extend_schema(
|
||||
tags=["Plant"],
|
||||
summary="ایجاد گیاه جدید",
|
||||
description="یک گیاه جدید با مشخصات دادهشده ایجاد میکند.",
|
||||
request=PlantSerializer,
|
||||
responses={
|
||||
201: PlantSerializer,
|
||||
400: OpenApiResponse(description="داده نامعتبر"),
|
||||
},
|
||||
examples=[
|
||||
OpenApiExample(
|
||||
"نمونه درخواست",
|
||||
value={
|
||||
"name": "گوجهفرنگی",
|
||||
"light": "آفتاب کامل",
|
||||
"watering": "منظم، هفتهای ۲-۳ بار",
|
||||
"soil": "لومی، غنی از مواد آلی",
|
||||
"temperature": "۲۰-۳۰ درجه سانتیگراد",
|
||||
"planting_season": "بهار",
|
||||
"harvest_time": "۷۰-۹۰ روز پس از کاشت",
|
||||
"spacing": "۴۵-۶۰ سانتیمتر",
|
||||
"fertilizer": "کود NPK متعادل",
|
||||
},
|
||||
request_only=True,
|
||||
),
|
||||
],
|
||||
)
|
||||
def post(self, request):
|
||||
serializer = PlantSerializer(data=request.data)
|
||||
if not serializer.is_valid():
|
||||
return Response(
|
||||
{"code": 400, "msg": "داده نامعتبر.", "data": serializer.errors},
|
||||
status=status.HTTP_400_BAD_REQUEST,
|
||||
)
|
||||
serializer.save()
|
||||
return Response(
|
||||
{"code": 201, "msg": "success", "data": serializer.data},
|
||||
status=status.HTTP_201_CREATED,
|
||||
)
|
||||
|
||||
|
||||
class PlantDetailView(APIView):
|
||||
"""دریافت، ویرایش و حذف یک گیاه."""
|
||||
|
||||
def _get_plant(self, pk):
|
||||
return Plant.objects.filter(pk=pk).first()
|
||||
|
||||
@extend_schema(
|
||||
tags=["Plant"],
|
||||
summary="جزئیات گیاه",
|
||||
description="مشخصات یک گیاه را بر اساس شناسه برمیگرداند.",
|
||||
responses={
|
||||
200: PlantSerializer,
|
||||
404: OpenApiResponse(description="گیاه یافت نشد"),
|
||||
},
|
||||
)
|
||||
def get(self, request, pk):
|
||||
plant = self._get_plant(pk)
|
||||
if not plant:
|
||||
return Response(
|
||||
{"code": 404, "msg": "گیاه یافت نشد.", "data": None},
|
||||
status=status.HTTP_404_NOT_FOUND,
|
||||
)
|
||||
serializer = PlantSerializer(plant)
|
||||
return Response(
|
||||
{"code": 200, "msg": "success", "data": serializer.data},
|
||||
status=status.HTTP_200_OK,
|
||||
)
|
||||
|
||||
@extend_schema(
|
||||
tags=["Plant"],
|
||||
summary="ویرایش کامل گیاه",
|
||||
description="تمام فیلدهای یک گیاه را آپدیت میکند.",
|
||||
request=PlantSerializer,
|
||||
responses={
|
||||
200: PlantSerializer,
|
||||
400: OpenApiResponse(description="داده نامعتبر"),
|
||||
404: OpenApiResponse(description="گیاه یافت نشد"),
|
||||
},
|
||||
)
|
||||
def put(self, request, pk):
|
||||
plant = self._get_plant(pk)
|
||||
if not plant:
|
||||
return Response(
|
||||
{"code": 404, "msg": "گیاه یافت نشد.", "data": None},
|
||||
status=status.HTTP_404_NOT_FOUND,
|
||||
)
|
||||
serializer = PlantSerializer(plant, data=request.data)
|
||||
if not serializer.is_valid():
|
||||
return Response(
|
||||
{"code": 400, "msg": "داده نامعتبر.", "data": serializer.errors},
|
||||
status=status.HTTP_400_BAD_REQUEST,
|
||||
)
|
||||
serializer.save()
|
||||
return Response(
|
||||
{"code": 200, "msg": "success", "data": serializer.data},
|
||||
status=status.HTTP_200_OK,
|
||||
)
|
||||
|
||||
@extend_schema(
|
||||
tags=["Plant"],
|
||||
summary="ویرایش جزئی گیاه",
|
||||
description="فقط فیلدهای ارسالشده آپدیت میشوند.",
|
||||
request=PlantSerializer,
|
||||
responses={
|
||||
200: PlantSerializer,
|
||||
400: OpenApiResponse(description="داده نامعتبر"),
|
||||
404: OpenApiResponse(description="گیاه یافت نشد"),
|
||||
},
|
||||
)
|
||||
def patch(self, request, pk):
|
||||
plant = self._get_plant(pk)
|
||||
if not plant:
|
||||
return Response(
|
||||
{"code": 404, "msg": "گیاه یافت نشد.", "data": None},
|
||||
status=status.HTTP_404_NOT_FOUND,
|
||||
)
|
||||
serializer = PlantSerializer(plant, data=request.data, partial=True)
|
||||
if not serializer.is_valid():
|
||||
return Response(
|
||||
{"code": 400, "msg": "داده نامعتبر.", "data": serializer.errors},
|
||||
status=status.HTTP_400_BAD_REQUEST,
|
||||
)
|
||||
serializer.save()
|
||||
return Response(
|
||||
{"code": 200, "msg": "success", "data": serializer.data},
|
||||
status=status.HTTP_200_OK,
|
||||
)
|
||||
|
||||
@extend_schema(
|
||||
tags=["Plant"],
|
||||
summary="حذف گیاه",
|
||||
description="یک گیاه را حذف میکند.",
|
||||
responses={
|
||||
200: OpenApiResponse(description="حذف موفق"),
|
||||
404: OpenApiResponse(description="گیاه یافت نشد"),
|
||||
},
|
||||
)
|
||||
def delete(self, request, pk):
|
||||
plant = self._get_plant(pk)
|
||||
if not plant:
|
||||
return Response(
|
||||
{"code": 404, "msg": "گیاه یافت نشد.", "data": None},
|
||||
status=status.HTTP_404_NOT_FOUND,
|
||||
)
|
||||
plant.delete()
|
||||
return Response(
|
||||
{"code": 200, "msg": "گیاه با موفقیت حذف شد.", "data": None},
|
||||
status=status.HTTP_200_OK,
|
||||
)
|
||||
|
||||
|
||||
class PlantFetchInfoView(APIView):
|
||||
"""دریافت مشخصات گیاه از API خارجی بر اساس نام."""
|
||||
|
||||
@extend_schema(
|
||||
tags=["Plant"],
|
||||
summary="دریافت مشخصات گیاه از API خارجی",
|
||||
description="بر اساس نام گیاه، مشخصات آن را از API خارجی دریافت میکند. (فعلاً خالی)",
|
||||
request=inline_serializer(
|
||||
name="PlantFetchInfoRequest",
|
||||
fields={
|
||||
"name": drf_serializers.CharField(help_text="نام گیاه"),
|
||||
},
|
||||
),
|
||||
responses={
|
||||
200: PlantSerializer,
|
||||
400: OpenApiResponse(description="نام گیاه ارسال نشده"),
|
||||
503: OpenApiResponse(description="سرویس در دسترس نیست"),
|
||||
},
|
||||
examples=[
|
||||
OpenApiExample(
|
||||
"نمونه درخواست",
|
||||
value={"name": "گوجهفرنگی"},
|
||||
request_only=True,
|
||||
),
|
||||
],
|
||||
)
|
||||
def post(self, request):
|
||||
plant_name = request.data.get("name")
|
||||
if not plant_name:
|
||||
return Response(
|
||||
{"code": 400, "msg": "نام گیاه الزامی است.", "data": None},
|
||||
status=status.HTTP_400_BAD_REQUEST,
|
||||
)
|
||||
|
||||
result = fetch_plant_info_from_api(plant_name)
|
||||
if result is None:
|
||||
return Response(
|
||||
{
|
||||
"code": 503,
|
||||
"msg": "سرویس API هنوز پیادهسازی نشده است.",
|
||||
"data": None,
|
||||
},
|
||||
status=status.HTTP_503_SERVICE_UNAVAILABLE,
|
||||
)
|
||||
|
||||
return Response(
|
||||
{"code": 200, "msg": "success", "data": result},
|
||||
status=status.HTTP_200_OK,
|
||||
)
|
||||
Reference in New Issue
Block a user