#!/usr/bin/env python3
"""
Configuration loader: load Pipeline configuration from a JSON file.
All configuration is stored in JSON; this file only handles loading and access helpers.
"""

import json
import os
from pathlib import Path
from typing import Optional, Dict, Any


class ConfigValue:
    """Descriptor for a scalar configuration value."""
    def __init__(self, key: str, default: Any = None):
        self.key = key
        self.default = default
    
    def __get__(self, obj, objtype=None):
        return objtype._get(self.key, self.default)


class ConfigPathValue:
    """Descriptor for a Path‑typed configuration value."""
    def __init__(self, key: str):
        self.key = key
    
    def __get__(self, obj, objtype=None):
        value = objtype._get(self.key)
        if value is None:
            return None
        if isinstance(value, str):
            return Path(value)
        return value


class Config:
    """Pipeline configuration class - loads settings from a JSON file."""
    
    # Configuration dictionary (loaded from JSON file)
    _config_dict: Dict[str, Any] = {}
    _loaded: bool = False
    
    @classmethod
    def _get_default_config_path(cls) -> Path:
        """Get the default configuration file path."""
        # Go up from src/utils/config.py to project root and locate config/config.json
        return Path(__file__).parent.parent.parent / "config" / "config.json"
    
    @classmethod
    def _load_default_config(cls):
        """Load configuration from the default JSON file."""
        default_config_path = cls._get_default_config_path()
        if default_config_path.exists():
            with open(default_config_path, "r", encoding="utf-8") as f:
                cls._config_dict = json.load(f)
       
        cls._loaded = True
    
    @classmethod
    def _ensure_loaded(cls):
        """Ensure the configuration has been loaded."""
        if not cls._loaded:
            cls._load_default_config()
    
    @classmethod
    def _get(cls, key: str, default: Any = None) -> Any:
        """Get a configuration value."""
        cls._ensure_loaded()
        # Environment variables take highest priority
        env_key = f"OPENAI_{key.upper()}"
        if key == "api_key":
            env_key = "OPENAI_API_KEY"
        elif key == "model":
            env_key = "OPENAI_MODEL"
        elif key == "base_url":
            env_key = "OPENAI_BASE_URL"
        
        env_value = os.getenv(env_key)
        if env_value is not None:
            return env_value
        
        # Fallback to the in‑memory configuration dictionary
        return cls._config_dict.get(key, default)
    
    # Configuration fields (read from JSON via descriptors)
    API_KEY = ConfigValue("api_key")
    MODEL = ConfigValue("model", "gpt-4o-mini")
    GENERATOR_MODEL = ConfigValue("generator_model")
    JUDGE_MODEL = ConfigValue("judge_model")
    PLOTTER_MODEL = ConfigValue("plotter_model")
    VL_MODEL = ConfigValue("vl_model")
    VISUALIZE_QA_MODEL = ConfigValue("visualize_qa_model")
    BASE_URL = ConfigValue("base_url", "https://api.openai.com/v1")
    DELAY = ConfigValue("delay", 1.0)
    MAX_RETRIES = ConfigValue("max_retries", 3)
    MAX_WORKERS = ConfigValue("max_workers", 1)
    AUTO_RESUME = ConfigValue("auto_resume", False)
    SKIP_IF_EXISTS = ConfigValue("skip_if_exists", True)
    FIGURES_DIR = ConfigPathValue("figures_dir")
    DEBUG = ConfigValue("debug", False)
    
    @classmethod
    def load_from_file(cls, config_file: Path):
        """Load configuration from the given JSON file."""
        if not config_file.exists():
            raise FileNotFoundError(f"Configuration file does not exist: {config_file}")
        
        if config_file.suffix != ".json":
            raise ValueError(f"Configuration file must be JSON (.json), got: {config_file}")
        
        with open(config_file, "r", encoding="utf-8") as f:
            cls._config_dict = json.load(f)
        cls._loaded = True
    
    @classmethod
    def to_dict(cls) -> dict:
        """Return the current configuration as a standard dictionary."""
        cls._ensure_loaded()
        return cls._config_dict.copy()
    