- apps/weather 新規作成(WeatherRecord モデル、5種APIエンドポイント) - GET /api/weather/records/ 日次データ一覧 - GET /api/weather/summary/ 月別・年間集計 - GET /api/weather/gdd/ 有効積算温度(GDD)計算 - GET /api/weather/similarity/ 類似年分析(開花・収穫予測の基礎) - POST /api/weather/sync/ Windmill向け日次更新(APIキー認証) - management command: fetch_weather(初回一括・差分取得) - Crop.base_temp フィールド追加(GDD基準温度、default=0.0℃) - docker-compose.yml: MAIL_API_KEY 環境変数を追加(ローカルテスト修正) - requirements.txt: requests>=2.31 追加 Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
45 lines
1.6 KiB
Python
45 lines
1.6 KiB
Python
from django.db import models
|
|
from apps.fields.models import Field
|
|
|
|
|
|
class Crop(models.Model):
|
|
name = models.CharField(max_length=100, unique=True, verbose_name="作物名")
|
|
base_temp = models.FloatField(default=0.0, verbose_name="有効積算温度 基準温度(℃)")
|
|
|
|
class Meta:
|
|
verbose_name = "作物マスタ"
|
|
verbose_name_plural = "作物マスタ"
|
|
|
|
def __str__(self):
|
|
return self.name
|
|
|
|
|
|
class Variety(models.Model):
|
|
crop = models.ForeignKey(Crop, on_delete=models.CASCADE, related_name='varieties', verbose_name="作物")
|
|
name = models.CharField(max_length=100, verbose_name="品種名")
|
|
|
|
class Meta:
|
|
verbose_name = "品種マスタ"
|
|
verbose_name_plural = "品種マスタ"
|
|
unique_together = [['crop', 'name']]
|
|
|
|
def __str__(self):
|
|
return f"{self.crop.name} - {self.name}"
|
|
|
|
|
|
class Plan(models.Model):
|
|
field = models.ForeignKey(Field, on_delete=models.CASCADE, related_name='plans', verbose_name="圃場")
|
|
year = models.IntegerField(verbose_name="作付年度")
|
|
crop = models.ForeignKey(Crop, on_delete=models.CASCADE, related_name='plans', verbose_name="作物")
|
|
variety = models.ForeignKey(Variety, on_delete=models.SET_NULL, related_name='plans', verbose_name="品種", blank=True, null=True)
|
|
notes = models.TextField(blank=True, null=True, verbose_name="備考")
|
|
|
|
class Meta:
|
|
verbose_name = "作付け計画"
|
|
verbose_name_plural = "作付け計画"
|
|
unique_together = [['field', 'year']]
|
|
ordering = ['-year', 'field']
|
|
|
|
def __str__(self):
|
|
return f"{self.field.name} - {self.year} - {self.crop.name}"
|