import json
from dataclasses import dataclass
from pathlib import Path
from typing import Any, Literal

@dataclass(frozen=True)
class DataGenConfig:
    experiment: str
    n_samples: int
    t_start: float
    t_end: float
    dt: float
    subsample_stride: int
    param_ranges: dict[str, Any]
    n_dim: int
    output_dir: Path

    n_timesteps: int

    sampling_mode: Literal["independent", "crops"] = "independent"
    crop_length: int | None = None      # in timesteps AFTER subsampling
    transient_cutoff: int | None = None # in timesteps AFTER subsampling
    parameter_mode: Literal["independent", "shared"] = "independent"
    parameters: list[float] | None = None

    crop_validator: dict[str, Any] | None = None

    @classmethod
    def from_json(cls, json_path: Path | str) -> "DataGenConfig":
        with open(json_path, 'r') as f:
            data = json.load(f)
            data['n_timesteps'] = int((data['t_end'] - data['t_start']) / data['dt'] / data['subsample_stride'])
        return cls(**data)
    
    def __post_init__(self):
        if self.t_end <= self.t_start:
            raise ValueError("t_end must be > t_start")
        if self.n_samples <= 0:
            raise ValueError("n_samples must be positive")
        if self.dt <= 0:
            raise ValueError("dt must be positive")
        if self.subsample_stride <= 0:
            raise ValueError("subsample_stride must be positive")
        if not self.param_ranges:
            raise ValueError("param_ranges cannot be empty")
        if not self.n_timesteps > 0:
            raise ValueError("n_timesteps must be positive")

        if self.sampling_mode not in {"independent", "crops"}:
            raise ValueError("sampling_mode must be 'independent' or 'crops'")

        if self.sampling_mode == "crops":
            if self.crop_length is None or self.crop_length <= 0:
                raise ValueError("crop_length required and must be positive for crops mode")
            if self.transient_cutoff is None or self.transient_cutoff < 0:
                raise ValueError("transient_cutoff required and must be non-negative for crops mode")

            usable = self.n_timesteps - self.transient_cutoff
            if usable <= self.crop_length:
                raise ValueError("Not enough timesteps after transient_cutoff for one crop")
