"""
Database schemas for agriculture_environmental_Eco_Agri_Data_Monitor

This module contains all database schema classes for the agriculture_environmental domain.
All classes are Pydantic BaseModel subclasses.

Generated by ClassDBAgent.
"""

import json
from datetime import date
from datetime import datetime
from datetime import datetime, date
from pathlib import Path
from pydantic import Field
from pydantic import PrivateAttr, Field
from pydantic import PrivateAttr, Field, field_validator
from scale_env.data_model.thread_safe_base import ThreadSafeBase, with_instance_key
from scale_env.environment.db import DB, DictAccessMixin as BaseModel
from typing import Dict, List, Any, Optional
from typing import Dict, List, Any, Optional, Literal

@with_instance_key("field_id")
class AgriculturalField(BaseModel, ThreadSafeBase["AgriculturalField"]):
    field_id: str = Field(..., description="Unique identifier of the agricultural field")
    field_name: Optional[str] = Field(default=None, description="Name of the agricultural field")
    area_hectares: Optional[float] = Field(default=None, description="Total area of the field in hectares")
    timestamp: datetime = Field(default_factory=datetime.now, description="Timestamp when field record was created")

@with_instance_key("reading_id")
class SoilMoistureReading(BaseModel, ThreadSafeBase["SoilMoistureReading"]):
    reading_id: str = Field(..., description="Unique identifier for the soil moisture reading")
    field_id: str = Field(..., description="Reference to agricultural field")
    latitude: float = Field(..., description="Latitude coordinate of measurement point")
    longitude: float = Field(..., description="Longitude coordinate of measurement point")
    depth_cm: float = Field(..., description="Depth of soil measurement in centimeters")
    moisture_percentage: float = Field(..., description="Soil moisture content as percentage")
    timestamp: datetime = Field(..., description="Time of measurement")

@with_instance_key("reading_id")
class AirTemperatureReading(BaseModel, ThreadSafeBase["AirTemperatureReading"]):
    reading_id: str = Field(..., description="Unique identifier for the temperature reading")
    field_id: str = Field(..., description="Reference to agricultural field")
    temperature_celsius: float = Field(..., description="Air temperature in degrees Celsius")
    relative_humidity: Optional[float] = Field(default=None, description="Relative humidity percentage")
    timestamp: datetime = Field(..., description="Time of measurement")

@with_instance_key("measurement_id")
class RainfallMeasurement(BaseModel, ThreadSafeBase["RainfallMeasurement"]):
    measurement_id: str = Field(default=..., description="Unique identifier for the rainfall measurement")
    field_id: str = Field(default=..., description="Reference to agricultural field")
    rainfall_mm: float = Field(default=..., description="Rainfall amount in millimeters")
    start_time: datetime = Field(default=..., description="Start time of rainfall period")
    end_time: datetime = Field(default=..., description="End time of rainfall period")
    duration_hours: Optional[float] = Field(default=None, description="Duration of rainfall period in hours")

@with_instance_key("measurement_id")
class SoilPhMeasurement(BaseModel, ThreadSafeBase["SoilPhMeasurement"]):
    measurement_id: str = Field(..., description="Unique identifier for the pH measurement")
    field_id: str = Field(..., description="Reference to agricultural field")
    latitude: float = Field(..., description="Latitude coordinate of measurement point")
    longitude: float = Field(..., description="Longitude coordinate of measurement point")
    depth_cm: float = Field(..., description="Depth of soil measurement in centimeters")
    ph_value: float = Field(..., description="Soil pH value (0-14 scale)")
    soil_acidity_status: Optional[str] = Field(None, description="Classification of soil acidity level")
    timestamp: datetime = Field(..., description="Time of measurement")

@with_instance_key("measurement_id")

class NpkNutrientMeasurement(BaseModel, ThreadSafeBase["NpkNutrientMeasurement"]):
    measurement_id: str = Field(..., description="Unique identifier for the NPK measurement")
    field_id: str = Field(..., description="Reference to agricultural field")
    nitrogen_ppm: float = Field(..., description="Nitrogen content in parts per million")
    phosphorus_ppm: float = Field(..., description="Phosphorus content in parts per million")
    potassium_ppm: float = Field(..., description="Potassium content in parts per million")
    latitude: float = Field(..., description="Latitude coordinate of measurement point")
    longitude: float = Field(..., description="Longitude coordinate of measurement point")
    nutrient_balance_status: Optional[str] = Field(None, description="Overall nutrient balance status")
    timestamp: datetime = Field(..., description="Time of measurement")

@with_instance_key("measurement_id")
class WindMeasurement(BaseModel, ThreadSafeBase["WindMeasurement"]):
    measurement_id: str = Field(..., description="Unique identifier for the wind measurement")
    field_id: str = Field(..., description="Reference to agricultural field")
    wind_speed_ms: float = Field(..., description="Wind speed in meters per second")
    wind_direction_degrees: float = Field(..., description="Wind direction in degrees (0-360)")
    wind_category: Optional[str] = Field(default=None, description="Beaufort scale wind category")
    timestamp: datetime = Field(..., description="Time of measurement")

@with_instance_key("measurement_id")
class SolarRadiationMeasurement(BaseModel, ThreadSafeBase["SolarRadiationMeasurement"]):      
    measurement_id: str = Field(default=..., description="Unique identifier for the radiation measurement")
    field_id: str = Field(default=..., description="Reference to agricultural field")
    radiation_wm2: float = Field(default=..., description="Solar radiation in watts per square meter")
    radiation_level: Optional[str] = Field(default=None, description="Categorized radiation intensity level")
    timestamp: datetime = Field(default=..., description="Time of measurement")

@with_instance_key("observation_id")
class PestObservation(BaseModel, ThreadSafeBase["PestObservation"]):      
    observation_id: str = Field(..., description="Unique identifier for the pest observation")
    field_id: str = Field(..., description="Reference to agricultural field")
    pest_type: str = Field(..., description="Type or species of pest observed")
    severity_level: Literal["low", "moderate", "high", "severe"] = Field(..., description="Severity level of infestation")
    area_hectares: float = Field(..., description="Estimated affected area in hectares")
    latitude: float = Field(..., description="Latitude coordinate of observation")
    longitude: float = Field(..., description="Longitude coordinate of observation")
    alert_triggered: Optional[bool] = Field(default=None, description="Whether an alert was triggered")
    timestamp: datetime = Field(..., description="Time of observation")

@with_instance_key("observation_id")
class DiseaseObservation(BaseModel, ThreadSafeBase["DiseaseObservation"]):
    observation_id: str = Field(..., description="Unique identifier for the disease observation")
    field_id: str = Field(..., description="Reference to agricultural field")
    disease_name: str = Field(..., description="Name of the disease observed")
    symptoms: List[str] = Field(..., description="JSON array of observed symptoms (list of strings)")
    infection_percentage: float = Field(..., description="Estimated percentage of plants infected")
    latitude: float = Field(..., description="Latitude coordinate of observation")
    longitude: float = Field(..., description="Longitude coordinate of observation")
    treatment_recommended: Optional[bool] = Field(None, description="Whether treatment is recommended")
    timestamp: datetime = Field(..., description="Time of observation")

    @field_validator("symptoms", mode="before")
    def parse_symptoms(cls, v):
        if isinstance(v, str):
            try:
                return json.loads(v)
            except Exception:
                raise ValueError("symptoms field must be a list or a JSON-encoded string representing a list.")
        return v

@with_instance_key("event_id")
class IrrigationEvent(BaseModel, ThreadSafeBase["IrrigationEvent"]):
    event_id: str = Field(..., description="Unique identifier for the irrigation event")
    field_id: str = Field(..., description="Reference to agricultural field")
    water_volume_liters: float = Field(..., description="Total water volume applied in liters")
    irrigation_method: Literal["drip", "sprinkler", "flood", "furrow", "center_pivot"] = Field(
        ..., description="Method of irrigation used"
    )
    start_time: datetime = Field(..., description="Start time of irrigation")
    end_time: datetime = Field(..., description="End time of irrigation")
    water_per_hectare: Optional[float] = Field(
        default=None, description="Water applied per hectare in liters"
    )

@with_instance_key("application_id")
class FertilizerApplication(BaseModel, ThreadSafeBase["FertilizerApplication"]):
    application_id: str = Field(..., description="Unique identifier for the fertilizer application")
    field_id: str = Field(..., description="Reference to agricultural field")
    fertilizer_type: str = Field(..., description="Type of fertilizer applied")
    amount_kg: float = Field(..., description="Amount of fertilizer applied in kilograms")
    application_method: Literal["broadcast", "band", "foliar", "fertigation", "injection"] = Field(
        ..., description="Method of fertilizer application"
    )
    npk_ratio: Optional[str] = Field(default=None, description="NPK ratio of the fertilizer")
    nitrogen_content_kg: Optional[float] = Field(default=None, description="Total nitrogen content applied in kg")
    timestamp: datetime = Field(..., description="Time of application")

@with_instance_key("measurement_id")
class CropYieldMeasurement(BaseModel, ThreadSafeBase["CropYieldMeasurement"]):
    measurement_id: str = Field(..., description="Unique identifier for the yield measurement")
    field_id: str = Field(..., description="Reference to agricultural field")
    crop_type: str = Field(..., description="Type of crop harvested")
    yield_kg: float = Field(..., description="Total yield in kilograms")
    area_hectares: float = Field(..., description="Harvested area in hectares")
    moisture_percentage: Optional[float] = Field(None, description="Moisture content of harvested crop")
    yield_per_hectare: Optional[float] = Field(None, description="Yield per hectare in kilograms")
    harvest_date: date = Field(..., description="Date of harvest")

@with_instance_key("measurement_id")
class CarbonFootprintMeasurement(BaseModel, ThreadSafeBase["CarbonFootprintMeasurement"]):
    measurement_id: str = Field(..., description="Unique identifier for the carbon measurement")
    field_id: str = Field(..., description="Reference to agricultural field")
    activity_type: Literal["tillage", "fertilization", "irrigation", "pesticide_application", "harvest", "transport"] = Field(
        ..., description="Type of agricultural activity"
    )
    co2_equivalent_kg: float = Field(..., description="CO2 equivalent emissions in kilograms")
    cumulative_co2_equivalent_kg: Optional[float] = Field(None, description="Cumulative CO2 equivalent emissions")
    activity_date: date = Field(..., description="Date of activity")

@with_instance_key("measurement_id")
class WaterQualityMeasurement(BaseModel, ThreadSafeBase["WaterQualityMeasurement"]):
    measurement_id: str = Field(..., description="Unique identifier for the water quality measurement")
    field_id: str = Field(..., description="Reference to agricultural field")
    water_source_type: Literal["well", "river", "reservoir", "canal", "recycled"] = Field(
        ..., description="Type of water source"
    )
    ph_value: float = Field(..., description="pH value of water")
    ec_ds_per_m: float = Field(..., description="Electrical conductivity in deciSiemens per meter")
    total_dissolved_solids_ppm: Optional[float] = Field(
        None, description="Total dissolved solids in parts per million"
    )
    water_quality_rating: Optional[str] = Field(
        None, description="Overall water quality rating"
    )
    timestamp: datetime = Field(..., description="Time of measurement")

@with_instance_key("observation_id")
class BiodiversityObservation(BaseModel, ThreadSafeBase["BiodiversityObservation"]):
    observation_id: str = Field(..., description="Unique identifier for the biodiversity observation")
    field_id: str = Field(..., description="Reference to agricultural field")
    species_name: str = Field(..., description="Name of observed species")
    species_category: Literal["pollinator", "beneficial_insect", "bird", "mammal", "plant", "microorganism"] = Field(
        ..., description="Category of species"
    )
    abundance_count: int = Field(..., description="Number of individuals observed")
    observation_method: Literal["visual_count", "trap", "transect", "quadrat", "camera_trap"] = Field(
        ..., description="Method used for observation"
    )
    species_status: Optional[str] = Field(None, description="Conservation status of species")
    timestamp: datetime = Field(..., description="Time of observation")

@with_instance_key("consumption_id")
class EnergyConsumption(BaseModel, ThreadSafeBase["EnergyConsumption"]):
    consumption_id: str = Field(..., description="Unique identifier for the energy consumption record")
    field_id: str = Field(..., description="Reference to agricultural field")
    energy_source: Literal["electricity", "diesel", "gasoline", "natural_gas", "solar", "wind"] = Field(
        ..., description="Source of energy consumed"
    )
    consumption_amount: float = Field(..., description="Amount of energy consumed")
    consumption_unit: Literal["kwh", "liters", "cubic_meters", "mj"] = Field(
        ..., description="Unit of energy consumption"
    )
    operation_type: Literal["tillage", "planting", "irrigation", "harvesting", "transport", "processing"] = Field(
        ..., description="Type of agricultural operation"
    )
    co2_equivalent_kg: Optional[float] = Field(
        None, description="CO2 emissions from energy consumption in kilograms"
    )
    timestamp: datetime = Field(..., description="Time of consumption")

@with_instance_key("forecast_id")
class WeatherForecast(BaseModel, ThreadSafeBase["WeatherForecast"]):
    forecast_id: str = Field(default=..., description="Unique identifier for the weather forecast")
    field_id: str = Field(default=..., description="Reference to agricultural field")
    forecast_date: date = Field(default=..., description="Date of forecast")
    max_temperature_celsius: float = Field(default=..., description="Forecasted maximum temperature")
    min_temperature_celsius: float = Field(default=..., description="Forecasted minimum temperature")
    precipitation_probability: float = Field(default=..., description="Probability of precipitation as percentage")
    rainfall_mm: Optional[float] = Field(default=None, description="Expected rainfall amount in millimeters")
    wind_speed_ms: Optional[float] = Field(default=None, description="Forecasted wind speed in meters per second")
    forecast_confidence: Optional[str] = Field(default=None, description="Confidence level of forecast")

@with_instance_key("measurement_id")
class SoilErosionMeasurement(BaseModel, ThreadSafeBase["SoilErosionMeasurement"]):
    measurement_id: str = Field(default=..., description="Unique identifier for the erosion measurement")
    field_id: str = Field(default=..., description="Reference to agricultural field")
    erosion_type: Literal["water_erosion", "wind_erosion", "tillage_erosion"] = Field(default=..., description="Type of soil erosion")
    soil_loss_tons_per_ha: float = Field(default=..., description="Estimated soil loss in tons per hectare")
    measurement_method: Literal["sediment_trap", "pin_measurement", "modeling", "visual_assessment"] = Field(default=..., description="Method used for erosion measurement")
    latitude: float = Field(default=..., description="Latitude coordinate of measurement point")
    longitude: float = Field(default=..., description="Longitude coordinate of measurement point")
    severity_level: Optional[str] = Field(default=None, description="Categorized erosion severity level")
    timestamp: datetime = Field(default=..., description="Time of measurement")

@with_instance_key("measurement_id")
class GroundwaterLevel(BaseModel, ThreadSafeBase["GroundwaterLevel"]):
    measurement_id: str = Field(..., description="Unique identifier for the groundwater level measurement")
    field_id: str = Field(..., description="Reference to agricultural field")
    well_id: str = Field(..., description="Identifier of the monitoring well")
    water_depth_meters: float = Field(..., description="Depth to water table in meters below surface")
    water_table_status: Optional[str] = Field(None, description="Status of water table level")
    timestamp: datetime = Field(..., description="Time of measurement")

@with_instance_key("measurement_id")
class LeafAreaIndexMeasurement(BaseModel, ThreadSafeBase["LeafAreaIndexMeasurement"]):
    measurement_id: str = Field(..., description="Unique identifier for the LAI measurement")
    field_id: str = Field(..., description="Reference to agricultural field")
    leaf_area_index: float = Field(..., description="Leaf area index value (m²/m²)")
    measurement_method: Literal[
        "direct_harvest",
        "optical_sensor",
        "hemispherical_photography",
        "remote_sensing"
    ] = Field(..., description="Method used for LAI measurement")
    latitude: float = Field(..., description="Latitude coordinate of measurement")
    longitude: float = Field(..., description="Longitude coordinate of measurement")
    canopy_coverage_status: Optional[str] = Field(None, description="Status of canopy coverage")
    timestamp: datetime = Field(..., description="Time of measurement")

@with_instance_key("observation_id")
class PollinatorActivity(BaseModel, ThreadSafeBase["PollinatorActivity"]):
    observation_id: str = Field(..., description="Unique identifier for the pollinator activity observation")
    field_id: str = Field(..., description="Reference to agricultural field")
    species_name: str = Field(..., description="Species of pollinator observed")
    visit_frequency_per_minute: float = Field(..., description="Number of flower visits per minute")
    observation_duration_minutes: int = Field(..., description="Duration of observation period in minutes")
    weather_conditions: Literal["sunny", "partly_cloudy", "cloudy", "light_rain", "windy"] = Field(..., description="Weather conditions during observation")
    activity_level: Optional[str] = Field(None, description="Categorized pollinator activity level")
    timestamp: datetime = Field(..., description="Time of observation")

class AgricultureEnvironmentalDB(DB):
    """Database containing all agriculture_environmental_Eco_Agri_Data_Monitor-related data"""
    agricultural_field: Optional[Dict[str, AgriculturalField]] = Field(
        default=None,
        description="Schema AgriculturalField"
    )
    soil_moisture_reading: Optional[Dict[str, SoilMoistureReading]] = Field(
        default=None,
        description="Schema SoilMoistureReading"
    )
    air_temperature_reading: Optional[Dict[str, AirTemperatureReading]] = Field(
        default=None,
        description="Schema AirTemperatureReading"
    )
    rainfall_measurement: Optional[Dict[str, RainfallMeasurement]] = Field(
        default=None,
        description="Schema RainfallMeasurement"
    )
    soil_ph_measurement: Optional[Dict[str, SoilPhMeasurement]] = Field(
        default=None,
        description="Schema SoilPhMeasurement"
    )
    npk_nutrient_measurement: Optional[Dict[str, NpkNutrientMeasurement]] = Field(
        default=None,
        description="Schema NpkNutrientMeasurement"
    )
    wind_measurement: Optional[Dict[str, WindMeasurement]] = Field(
        default=None,
        description="Schema WindMeasurement"
    )
    solar_radiation_measurement: Optional[Dict[str, SolarRadiationMeasurement]] = Field(
        default=None,
        description="Schema SolarRadiationMeasurement"
    )
    pest_observation: Optional[Dict[str, PestObservation]] = Field(
        default=None,
        description="Schema PestObservation"
    )
    disease_observation: Optional[Dict[str, DiseaseObservation]] = Field(
        default=None,
        description="Schema DiseaseObservation"
    )
    irrigation_event: Optional[Dict[str, IrrigationEvent]] = Field(
        default=None,
        description="Schema IrrigationEvent"
    )
    fertilizer_application: Optional[Dict[str, FertilizerApplication]] = Field(
        default=None,
        description="Schema FertilizerApplication"
    )
    crop_yield_measurement: Optional[Dict[str, CropYieldMeasurement]] = Field(
        default=None,
        description="Schema CropYieldMeasurement"
    )
    carbon_footprint_measurement: Optional[Dict[str, CarbonFootprintMeasurement]] = Field(
        default=None,
        description="Schema CarbonFootprintMeasurement"
    )
    water_quality_measurement: Optional[Dict[str, WaterQualityMeasurement]] = Field(
        default=None,
        description="Schema WaterQualityMeasurement"
    )
    biodiversity_observation: Optional[Dict[str, BiodiversityObservation]] = Field(
        default=None,
        description="Schema BiodiversityObservation"
    )
    energy_consumption: Optional[Dict[str, EnergyConsumption]] = Field(
        default=None,
        description="Schema EnergyConsumption"
    )
    weather_forecast: Optional[Dict[str, WeatherForecast]] = Field(
        default=None,
        description="Schema WeatherForecast"
    )
    soil_erosion_measurement: Optional[Dict[str, SoilErosionMeasurement]] = Field(
        default=None,
        description="Schema SoilErosionMeasurement"
    )
    groundwater_level: Optional[Dict[str, GroundwaterLevel]] = Field(
        default=None,
        description="Schema GroundwaterLevel"
    )
    leaf_area_index_measurement: Optional[Dict[str, LeafAreaIndexMeasurement]] = Field(
        default=None,
        description="Schema LeafAreaIndexMeasurement"
    )
    pollinator_activity: Optional[Dict[str, PollinatorActivity]] = Field(
        default=None,
        description="Schema PollinatorActivity"
    )