import os
from pathlib import Path
from typing import Dict, Optional
from src.model_loading.common.enums.model_enums import BitPrecision
from src.model_loading.common.models.identifier import ModelIdentifier
from src.model_loading.common.models.paths import ModelPaths
from src.model_loading.registry.models import Models

def get_model_path(path):
    value = os.environ.get(path)
    if not value:
        raise ValueError(f"Required environment variable '{path}' is missing or empty in .env file")
    return value

base_model_save_path = get_model_path('HF_HUB_PATH')

class ModelRegistry:
    """Registry for model paths and configurations"""
    def __init__(self, model_save_path: Optional[str] = base_model_save_path):
        self.model_save_path = Path(model_save_path) 
        self._model_paths: Dict[ModelIdentifier, ModelPaths] = {}
        self._initialize_models()
        
    def _initialize_models(self):
        """Initialize all model configurations"""
        self._initialize_base_models()
        self._initialize_tinyllama_models()
        self._initialize_llama2_models()
        self._initialize_llama3_models()
        self._initialize_llama31_models()
        self._initialize_llama32_models()
        self._initialize_opt_base_models()
        self._initialize_opt_huggingface_models()
        
    def _initialize_base_models(self):
        """Initialize all base model configurations"""
        self._register_model(Models.BaseModels.BLOOMZ, "bigscience/bloomz-1b1")
        self._register_model(Models.BaseModels.GPT2_LARGE, "openai-community/gpt2-large")
        
    def _initialize_tinyllama_models(self):
        """Initialize all TinyLlama model configurations"""
        # Base Models
        self._register_model(Models.TinyLlama.Base.CHAT, "TinyLlama/TinyLlama-1.1B-Chat-v1.0")
        self._register_model(Models.TinyLlama.Base.BASE, "TinyLlama/TinyLlama_v1.1")
        
        # HuggingFace Models
        tinyllama_base_path = "TinyLlama/TinyLlama-1.1B-Chat-v1.0"
        self._register_model(
            Models.TinyLlama.HuggingFace.BNB_8BIT,
            "PrunaAI/TinyLlama-TinyLlama-1.1B-Chat-v0.3-bnb-8bit-smashed",
            tinyllama_base_path
        )
        self._register_model(
            Models.TinyLlama.HuggingFace.BNB_4BIT,
            "PrunaAI/TinyLlama-TinyLlama-1.1B-Chat-v0.3-bnb-4bit-smashed",
            tinyllama_base_path
        )
        self._register_model(
            Models.TinyLlama.HuggingFace.AWQ_4BIT,
            "TheBloke/TinyLlama-1.1B-Chat-v0.3-AWQ",
            tinyllama_base_path
        )
        self._register_model(
            Models.TinyLlama.HuggingFace.GPTQ_8BIT,
            "PrunaAI/TinyLlama-TinyLlama-1.1B-Chat-v0.3-GPTQ-8bit-smashed",
            tinyllama_base_path
        )

    def _initialize_llama2_models(self):
        """Initialize Llama-2 models"""
        # Base Models
        self._register_model(
            Models.Llama2.Base.LLAMA_2_7B,
            "meta-llama/Llama-2-7b-hf"
        )

    def _initialize_llama3_models(self):
        """Initialize Llama-3 models"""
        self._initialize_llama3_base_models()
        self._initialize_llama3_huggingface_models()
        self._initialize_llama3_local_models()
        
    def _initialize_llama3_base_models(self):
        """Initialize Llama-3 base models"""
        self._register_model(
            Models.Llama3.Base.LLAMA_3_8B,
            "meta-llama/Meta-Llama-3-8B"
        )
        self._register_model(
            Models.Llama3.Base.LLAMA_3_70B,
            "meta-llama/Meta-Llama-3-70B"
        )

    def _initialize_llama3_huggingface_models(self):
        """Initialize Llama-3 HuggingFace models"""
        # 8B Models
        self._initialize_llama3_huggingface_8b_models()
        # 70B Models
        self._initialize_llama3_huggingface_70b_models()
    
    def _initialize_llama3_huggingface_8b_models(self):
        """Initialize Llama-3 HuggingFace 8B models"""
        base_path = "meta-llama/Meta-Llama-3-8B"
        
        # AQLM models
        self._register_model(
            Models.Llama3.HuggingFace.EightB.AQLM_2BIT,
            "ISTA-DASLab/Meta-Llama-3-8B-AQLM-2Bit-1x16",
            base_path
        )
        self._register_model(
            Models.Llama3.HuggingFace.EightB.AQLM_PV_2BIT,
            "ISTA-DASLab/Meta-Llama-3-8B-AQLM-PV-2Bit-1x16",
            base_path
        )
        self._register_model(
            Models.Llama3.HuggingFace.EightB.AQLM_PV_1BIT,
            "ISTA-DASLab/Meta-Llama-3-8B-AQLM-PV-1Bit-1x16",
            base_path
        )

        # AWQ models
        self._register_model(
            Models.Llama3.HuggingFace.EightB.AWQ_4BIT,
            "PrunaAI/meta-llama-Meta-Llama-3-8B-AWQ-4bit-smashed",
            base_path
        )
        
        # BNB models
        for model_config in [
            Models.Llama3.HuggingFace.EightB.BNB_8BIT,
            Models.Llama3.HuggingFace.EightB.BNB_4BIT
        ]:
            self._register_model(model_config, base_path)

        # GPTQ models
        for model_config in [
            Models.Llama3.HuggingFace.EightB.GPTQ_8BIT,
            Models.Llama3.HuggingFace.EightB.GPTQ_4BIT,
            Models.Llama3.HuggingFace.EightB.GPTQ_3BIT,
            Models.Llama3.HuggingFace.EightB.GPTQ_2BIT
        ]:
            self._register_model(model_config, base_path)

        # HQQ models
        for model_config in [
            Models.Llama3.HuggingFace.EightB.HQQ_8BIT,
            Models.Llama3.HuggingFace.EightB.HQQ_4BIT,
            Models.Llama3.HuggingFace.EightB.HQQ_3BIT,
            Models.Llama3.HuggingFace.EightB.HQQ_2BIT,
            Models.Llama3.HuggingFace.EightB.HQQ_1BIT
        ]:
            self._register_model(model_config, base_path)

        # QUANTO models
        for model_config in [
            Models.Llama3.HuggingFace.EightB.QUANTO_8BIT,
            Models.Llama3.HuggingFace.EightB.QUANTO_4BIT,
            Models.Llama3.HuggingFace.EightB.QUANTO_2BIT
        ]:
            self._register_model(model_config, base_path)

        # QoQ models
        self._register_model(
            Models.Llama3.HuggingFace.EightB.QOQ_4BIT,
            "mit-han-lab/Llama-3-8B-QServe",
            "mit-han-lab/Llama-3-8B-QServe"
        )

        # QuaRot models
        self._register_model(
            Models.Llama3.HuggingFace.EightB.QUAROT_8BIT,
            "AminBH/Llama-3-8B-Quarot-wiki-8bit",
            "AminBH/Llama-3-8B-Quarot-wiki-8bit"
        )
        self._register_model(
            Models.Llama3.HuggingFace.EightB.QUAROT_4BIT,
            "AminBH/Llama-3-8B-Quarot-wiki-4bit",
            "AminBH/Llama-3-8B-Quarot-wiki-4bit"
        )
        
    def _initialize_llama3_huggingface_70b_models(self):
        """Initialize Llama-3 HuggingFace 70B models"""
        base_path = "meta-llama/Meta-Llama-3-70B"
        
        # AWQ models
        self._register_model(
            Models.Llama3.HuggingFace.SeventyB.AWQ_4BIT,
            "TechxGenus/Meta-Llama-3-70B-AWQ",
            "TechxGenus/Meta-Llama-3-70B-AWQ"
        )
        
        # # GPTQ models
        # self._register_model(
        #     Models.Llama3.HuggingFace.SeventyB.GPTQ_4BIT,
        #     "TechxGenus/Meta-Llama-3-70B-GPTQ",
        #     "TechxGenus/Meta-Llama-3-70B-GPTQ"
        # )
        
        # AQLM models
        self._register_model(
            Models.Llama3.HuggingFace.SeventyB.AQLM_2BIT,
            "ISTA-DASLab/Meta-Llama-3-70B-AQLM-2Bit-1x16",
            base_path
        )
        self._register_model(
            Models.Llama3.HuggingFace.SeventyB.AQLM_PV_2BIT,
            "ISTA-DASLab/Meta-Llama-3-70B-AQLM-PV-2Bit-1x16",
            base_path
        )
        self._register_model(
            Models.Llama3.HuggingFace.SeventyB.AQLM_PV_1BIT,
            "ISTA-DASLab/Meta-Llama-3-70B-AQLM-PV-1Bit-1x16",
            base_path
        )
        
        # BNB models
        for model_config in [
            Models.Llama3.HuggingFace.SeventyB.BNB_8BIT,
            Models.Llama3.HuggingFace.SeventyB.BNB_4BIT
        ]:
            self._register_model(model_config, base_path)

        # GPTQ models
        for model_config in [
            Models.Llama3.HuggingFace.SeventyB.GPTQ_8BIT,
            Models.Llama3.HuggingFace.SeventyB.GPTQ_4BIT,
            Models.Llama3.HuggingFace.SeventyB.GPTQ_3BIT,
            Models.Llama3.HuggingFace.SeventyB.GPTQ_2BIT
        ]:
            self._register_model(model_config, base_path)

        # HQQ models
        for model_config in [
            Models.Llama3.HuggingFace.SeventyB.HQQ_8BIT,
            Models.Llama3.HuggingFace.SeventyB.HQQ_4BIT,
            Models.Llama3.HuggingFace.SeventyB.HQQ_3BIT,
            Models.Llama3.HuggingFace.SeventyB.HQQ_2BIT,
            Models.Llama3.HuggingFace.SeventyB.HQQ_1BIT
        ]:
            self._register_model(model_config, base_path)

        # QUANTO models
        for model_config in [
            Models.Llama3.HuggingFace.SeventyB.QUANTO_8BIT,
            Models.Llama3.HuggingFace.SeventyB.QUANTO_4BIT,
            Models.Llama3.HuggingFace.SeventyB.QUANTO_2BIT
        ]:
            self._register_model(model_config, base_path)
        
    def _initialize_llama3_local_models(self):
        """Initialize Llama-3 local models"""
        base_path = "meta-llama/Meta-Llama-3-8B"
        
        # QUANTO models
        self._register_model(
            Models.Llama3.Local.EightB.QUANTO_8BIT,
            os.path.join(self.model_save_path, "Meta-Llama-3-8B-QUANTO-8"),
            base_path
        )
        self._register_model(
            Models.Llama3.Local.EightB.QUANTO_8BIT_CALIB,
            os.path.join(self.model_save_path, "Meta-Llama-3-8B-QUANTO-CALIB-8"),
            base_path
        )
        self._register_model(
            Models.Llama3.Local.EightB.QUANTO_MIXED,
            os.path.join(self.model_save_path, "Meta-Llama-3-8B-QUANTO-8-mixed"),
            base_path
        )

        # AWQ models
        self._register_model(
            Models.Llama3.Local.EightB.AWQ_4BIT,
            os.path.join(self.model_save_path, "Meta-Llama-3-8B-AWQ-4bit"),
            base_path
        )

        # BNB models
        self._register_model(
            Models.Llama3.Local.EightB.BNB_8BIT,
            os.path.join(self.model_save_path, "Meta-Llama-3-8B-BNB-8"),
            base_path
        )
        self._register_model(
            Models.Llama3.Local.EightB.BNB_4BIT,
            os.path.join(self.model_save_path, "Meta-Llama-3-8B-BNB-4"),
            base_path
        )

        # HQQ models
        self._register_model(
            Models.Llama3.Local.EightB.HQQ_8BIT_UNIFORM,
            os.path.join(self.model_save_path, "Meta-Llama-3-8B-HQQ-8-uniform"),
            base_path
        )
        self._register_model(
            Models.Llama3.Local.EightB.HQQ_MIXED,
            os.path.join(self.model_save_path, "Meta-Llama-3-8B-HQQ-mixed"),
            base_path
        )
        self._register_model(
            Models.Llama3.Local.EightB.HQQ_LORA,
            os.path.join(self.model_save_path, "Meta-Llama-3-8B-HQQ-LORA"),
            base_path
        )
        
    def _initialize_llama31_models(self):
        """Initialize Llama-3.1 models"""
        self._initialize_llama31_base_models()
        self._initialize_llama31_huggingface_models()
        self._initialize_llama31_local_models()

    def _initialize_llama31_base_models(self):
        """Initialize Llama-3.1 base models"""
        self._register_model(
            Models.Llama31.Base.LLAMA_31_8B,
            "meta-llama/Llama-3.1-8B"
        )
        self._register_model(
            Models.Llama31.Base.LLAMA_31_70B,
            "meta-llama/Llama-3.1-70B"
        )

    def _initialize_llama31_huggingface_models(self):
        """Initialize Llama-3.1 HuggingFace models"""
        self._initialize_llama31_huggingface_8b_models()

    def _initialize_llama31_huggingface_8b_models(self):
        """Initialize Llama-3.1 HuggingFace 8B models"""
        base_path = "meta-llama/Llama-3.1-8B"
        
        # AWQ models
        self._register_model(
            Models.Llama31.HuggingFace.EightB.AWQ_4BIT,
            "JonathanBenaharon/Llama-3.1-8B-AWQ-4bit",
            "JonathanBenaharon/Llama-3.1-8B-AWQ-4bit"
        )

        # HQQ models
        for model_config in [
            Models.Llama31.HuggingFace.EightB.HQQ_8BIT,
            Models.Llama31.HuggingFace.EightB.HQQ_4BIT,
            Models.Llama31.HuggingFace.EightB.HQQ_3BIT,
            Models.Llama31.HuggingFace.EightB.HQQ_2BIT,
            Models.Llama31.HuggingFace.EightB.HQQ_1BIT
        ]:
            self._register_model(model_config, base_path)

        # QUANTO models
        for model_config in [
            Models.Llama31.HuggingFace.EightB.QUANTO_8BIT,
            Models.Llama31.HuggingFace.EightB.QUANTO_4BIT,
            Models.Llama31.HuggingFace.EightB.QUANTO_2BIT
        ]:
            self._register_model(model_config, base_path)

        # BNB models
        for model_config in [
            Models.Llama31.HuggingFace.EightB.BNB_8BIT,
            Models.Llama31.HuggingFace.EightB.BNB_4BIT
        ]:
            self._register_model(model_config, base_path)

        # GPTQ models
        for model_config in [
            Models.Llama31.HuggingFace.EightB.GPTQ_8BIT,
            Models.Llama31.HuggingFace.EightB.GPTQ_4BIT,
            Models.Llama31.HuggingFace.EightB.GPTQ_3BIT,
            Models.Llama31.HuggingFace.EightB.GPTQ_2BIT
        ]:
            self._register_model(model_config, base_path)
            
        # AQLM PV models
        self._register_model(
            Models.Llama31.HuggingFace.EightB.AQLM_PV_2BIT,
            "ISTA-DASLab/Meta-Llama-3.1-8B-AQLM-PV-2Bit-1x16-hf",
            "ISTA-DASLab/Meta-Llama-3.1-8B-AQLM-PV-2Bit-1x16-hf"
        )
        self._register_model(
            Models.Llama31.HuggingFace.EightB.AQLM_PV_1BIT,
            "ISTA-DASLab/Meta-Llama-3.1-8B-AQLM-PV-1Bit-1x16-hf",
            "ISTA-DASLab/Meta-Llama-3.1-8B-AQLM-PV-1Bit-1x16-hf"
        )

    def _initialize_llama31_local_models(self):
        """Initialize Llama-3.1 local models"""
        base_path = "meta-llama/Llama-3.1-8B"
        
        # AWQ models
        self._register_model(
            Models.Llama31.Local.EightB.AWQ_4BIT,
            os.path.join(self.model_save_path, "Llama-31-8B-AWQ-4bit"),
            base_path
        )
        
    def _initialize_llama32_models(self):
        """Initialize Llama-3.2 models"""
        self._initialize_llama32_base_models()
        self._initialize_llama32_huggingface_models()
        self._initialize_llama32_local_models()

    def _initialize_llama32_base_models(self):
        """Initialize Llama-3.2 base models"""
        self._register_model(
            Models.Llama32.Base.LLAMA_32_1B,
            "meta-llama/Llama-3.2-1B"
        )
        self._register_model(
            Models.Llama32.Base.LLAMA_32_3B,
            "meta-llama/Llama-3.2-3B"
        )

    def _initialize_llama32_huggingface_models(self):
        """Initialize Llama-3.2 HuggingFace models"""
        self._initialize_llama32_huggingface_1b_models()
        self._initialize_llama32_huggingface_3b_models()

    def _initialize_llama32_huggingface_1b_models(self):
        """Initialize Llama-3.2 HuggingFace 1B models"""
        base_path = "meta-llama/Llama-3.2-1B"

        # AWQ models
        self._register_model(
            Models.Llama32.HuggingFace.OneB.AWQ_4BIT,
            "skshmjn/Llama-3.2-1B-awq-4-bit",
            "skshmjn/Llama-3.2-1B-awq-4-bit"
        )

        # BNB models
        for model_config in [
            Models.Llama32.HuggingFace.OneB.BNB_8BIT,
            Models.Llama32.HuggingFace.OneB.BNB_4BIT
        ]:
            self._register_model(model_config, base_path)

        # GPTQ models
        for model_config in [
            Models.Llama32.HuggingFace.OneB.GPTQ_8BIT,
            Models.Llama32.HuggingFace.OneB.GPTQ_4BIT,
            Models.Llama32.HuggingFace.OneB.GPTQ_3BIT,
            Models.Llama32.HuggingFace.OneB.GPTQ_2BIT
        ]:
            self._register_model(model_config, base_path)

        # HQQ models
        for model_config in [
            Models.Llama32.HuggingFace.OneB.HQQ_8BIT,
            Models.Llama32.HuggingFace.OneB.HQQ_4BIT,
            Models.Llama32.HuggingFace.OneB.HQQ_3BIT,
            Models.Llama32.HuggingFace.OneB.HQQ_2BIT,
            Models.Llama32.HuggingFace.OneB.HQQ_1BIT
        ]:
            self._register_model(model_config, base_path)

        # QUANTO models
        for model_config in [
            Models.Llama32.HuggingFace.OneB.QUANTO_8BIT,
            Models.Llama32.HuggingFace.OneB.QUANTO_4BIT,
            Models.Llama32.HuggingFace.OneB.QUANTO_2BIT
        ]:
            self._register_model(model_config, base_path)
        
        # AQLM PV models
        self._register_model(
            Models.Llama32.HuggingFace.OneB.AQLM_PV_2BIT,
            "ISTA-DASLab/Llama-3.2-1B-AQLM-PV-2Bit-2x8",
            "ISTA-DASLab/Llama-3.2-1B-AQLM-PV-2Bit-2x8"
        )

    def _initialize_llama32_huggingface_3b_models(self):
        """Initialize Llama-3.2 HuggingFace 3B models"""
        base_path = "meta-llama/Llama-3.2-3B"

        # HQQ models
        for model_config in [
            Models.Llama32.HuggingFace.ThreeB.HQQ_8BIT,
            Models.Llama32.HuggingFace.ThreeB.HQQ_4BIT,
            Models.Llama32.HuggingFace.ThreeB.HQQ_3BIT,
            Models.Llama32.HuggingFace.ThreeB.HQQ_2BIT,
            Models.Llama32.HuggingFace.ThreeB.HQQ_1BIT
        ]:
            self._register_model(model_config, base_path)

        # QUANTO models
        for model_config in [
            Models.Llama32.HuggingFace.ThreeB.QUANTO_8BIT,
            Models.Llama32.HuggingFace.ThreeB.QUANTO_4BIT,
            Models.Llama32.HuggingFace.ThreeB.QUANTO_2BIT
        ]:
            self._register_model(model_config, base_path)

        # BNB models
        for model_config in [
            Models.Llama32.HuggingFace.ThreeB.BNB_8BIT,
            Models.Llama32.HuggingFace.ThreeB.BNB_4BIT
        ]:
            self._register_model(model_config, base_path)

        # GPTQ models
        for model_config in [
            Models.Llama32.HuggingFace.ThreeB.GPTQ_8BIT,
            Models.Llama32.HuggingFace.ThreeB.GPTQ_4BIT,
            Models.Llama32.HuggingFace.ThreeB.GPTQ_3BIT,
            Models.Llama32.HuggingFace.ThreeB.GPTQ_2BIT
        ]:
            self._register_model(model_config, base_path)
            
        # AQLM PV models
        self._register_model(
            Models.Llama32.HuggingFace.ThreeB.AQLM_PV_2BIT,
            "ISTA-DASLab/Llama-3.2-3B-AQLM-PV-2Bit-2x8",
            "ISTA-DASLab/Llama-3.2-3B-AQLM-PV-2Bit-2x8"
        )

    def _initialize_llama32_local_models(self):
        """Initialize Llama-3.2 local models"""
        # 1B models
        self._register_model(
            Models.Llama32.Local.OneB.AWQ_4BIT,
            os.path.join(self.model_save_path, "Llama-32-1B-AWQ-4bit"),
            "meta-llama/Llama-3.2-1B"
        )

        # 3B models
        self._register_model(
            Models.Llama32.Local.ThreeB.AWQ_4BIT,
            os.path.join(self.model_save_path, "Llama-32-3B-AWQ-4bit"),
            "meta-llama/Llama-3.2-3B"
        )
        
    def _initialize_opt_base_models(self):
        """Initialize OPT models"""
        # Base Models
        self._register_model(
            Models.Opt.Base.OPT_125M,
            "facebook/opt-125m"
        )
        self._register_model(
            Models.Opt.Base.OPT_350M,
            "facebook/opt-350m"
        )
        self._register_model(
            Models.Opt.Base.OPT_1B,
            "facebook/opt-1.3b"
        )
        self._register_model(
            Models.Opt.Base.OPT_2B,
            "facebook/opt-2.7b"
        )
        self._register_model(
            Models.Opt.Base.OPT_6B,
            "facebook/opt-6.7b"
        )
        self._register_model(
            Models.Opt.Base.OPT_13B,
            "facebook/opt-13b"
        )
    
    def _initialize_opt_huggingface_models(self):
        """Initialize OPT HuggingFace models"""
        # BNB models
        for model_config in [
            Models.Opt.HuggingFace.M125.BNB_8BIT,
            Models.Opt.HuggingFace.M125.BNB_4BIT,
        ]:
            self._register_model(model_config, "facebook/opt-125m")
        for model_config in [
            Models.Opt.HuggingFace.M350.BNB_8BIT,
            Models.Opt.HuggingFace.M350.BNB_4BIT,
        ]:
            self._register_model(model_config, "facebook/opt-350m")
        for model_config in [
            Models.Opt.HuggingFace.B1_3.BNB_8BIT,
            Models.Opt.HuggingFace.B1_3.BNB_4BIT,
        ]:
            self._register_model(model_config, "facebook/opt-1.3b")
        for model_config in [
            Models.Opt.HuggingFace.B2_7.BNB_8BIT,
            Models.Opt.HuggingFace.B2_7.BNB_4BIT,
        ]:
            self._register_model(model_config, "facebook/opt-2.7b")
        for model_config in [   
            Models.Opt.HuggingFace.B6_7.BNB_8BIT,
            Models.Opt.HuggingFace.B6_7.BNB_4BIT,
        ]:
            self._register_model(model_config, "facebook/opt-6.7b")
        for model_config in [
            Models.Opt.HuggingFace.B13.BNB_8BIT,
            Models.Opt.HuggingFace.B13.BNB_4BIT,
        ]:
            self._register_model(model_config, "facebook/opt-13b")
            
        # GPTQ models
        for model_config in [
            Models.Opt.HuggingFace.M125.GPTQ_8BIT,
            Models.Opt.HuggingFace.M125.GPTQ_4BIT,
            Models.Opt.HuggingFace.M125.GPTQ_3BIT,
            Models.Opt.HuggingFace.M125.GPTQ_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-125m")
        for model_config in [
            Models.Opt.HuggingFace.M350.GPTQ_8BIT,
            Models.Opt.HuggingFace.M350.GPTQ_4BIT,
            Models.Opt.HuggingFace.M350.GPTQ_3BIT,
            Models.Opt.HuggingFace.M350.GPTQ_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-350m")
        for model_config in [
            Models.Opt.HuggingFace.B1_3.GPTQ_8BIT,
            Models.Opt.HuggingFace.B1_3.GPTQ_4BIT,
            Models.Opt.HuggingFace.B1_3.GPTQ_3BIT,
            Models.Opt.HuggingFace.B1_3.GPTQ_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-1.3b")
        for model_config in [
            Models.Opt.HuggingFace.B2_7.GPTQ_8BIT,
            Models.Opt.HuggingFace.B2_7.GPTQ_4BIT,
            Models.Opt.HuggingFace.B2_7.GPTQ_3BIT,
            Models.Opt.HuggingFace.B2_7.GPTQ_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-2.7b")
        for model_config in [
            Models.Opt.HuggingFace.B6_7.GPTQ_8BIT,
            Models.Opt.HuggingFace.B6_7.GPTQ_4BIT,
            Models.Opt.HuggingFace.B6_7.GPTQ_3BIT,
            Models.Opt.HuggingFace.B6_7.GPTQ_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-6.7b")
        for model_config in [
            Models.Opt.HuggingFace.B13.GPTQ_8BIT,
            Models.Opt.HuggingFace.B13.GPTQ_4BIT,
            Models.Opt.HuggingFace.B13.GPTQ_3BIT,
            Models.Opt.HuggingFace.B13.GPTQ_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-13b")
        
        # HQQ models
        for model_config in [
            Models.Opt.HuggingFace.M125.HQQ_8BIT,
            Models.Opt.HuggingFace.M125.HQQ_4BIT,
            Models.Opt.HuggingFace.M125.HQQ_3BIT,
            Models.Opt.HuggingFace.M125.HQQ_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-125m")
        for model_config in [
            Models.Opt.HuggingFace.M350.HQQ_8BIT,
            Models.Opt.HuggingFace.M350.HQQ_4BIT,
            Models.Opt.HuggingFace.M350.HQQ_3BIT,
            Models.Opt.HuggingFace.M350.HQQ_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-350m")
        for model_config in [
            Models.Opt.HuggingFace.B1_3.HQQ_8BIT,
            Models.Opt.HuggingFace.B1_3.HQQ_4BIT,
            Models.Opt.HuggingFace.B1_3.HQQ_3BIT,
            Models.Opt.HuggingFace.B1_3.HQQ_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-1.3b")
        for model_config in [
            Models.Opt.HuggingFace.B2_7.HQQ_8BIT,
            Models.Opt.HuggingFace.B2_7.HQQ_4BIT,
            Models.Opt.HuggingFace.B2_7.HQQ_3BIT,
            Models.Opt.HuggingFace.B2_7.HQQ_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-2.7b")
        for model_config in [
            Models.Opt.HuggingFace.B6_7.HQQ_8BIT,
            Models.Opt.HuggingFace.B6_7.HQQ_4BIT,
            Models.Opt.HuggingFace.B6_7.HQQ_3BIT,
            Models.Opt.HuggingFace.B6_7.HQQ_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-6.7b")
        for model_config in [
            Models.Opt.HuggingFace.B13.HQQ_8BIT,
            Models.Opt.HuggingFace.B13.HQQ_4BIT,
            Models.Opt.HuggingFace.B13.HQQ_3BIT,
            Models.Opt.HuggingFace.B13.HQQ_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-13b")
        
        # Quanto models
        for model_config in [
            Models.Opt.HuggingFace.M125.QUANTO_8BIT,
            Models.Opt.HuggingFace.M125.QUANTO_4BIT,
            Models.Opt.HuggingFace.M125.QUANTO_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-125m")
        for model_config in [
            Models.Opt.HuggingFace.M350.QUANTO_8BIT,
            Models.Opt.HuggingFace.M350.QUANTO_4BIT,
            Models.Opt.HuggingFace.M350.QUANTO_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-350m")
        for model_config in [
            Models.Opt.HuggingFace.B1_3.QUANTO_8BIT,
            Models.Opt.HuggingFace.B1_3.QUANTO_4BIT,
            Models.Opt.HuggingFace.B1_3.QUANTO_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-1.3b")
        for model_config in [
            Models.Opt.HuggingFace.B2_7.QUANTO_8BIT,
            Models.Opt.HuggingFace.B2_7.QUANTO_4BIT,
            Models.Opt.HuggingFace.B2_7.QUANTO_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-2.7b")
        for model_config in [
            Models.Opt.HuggingFace.B6_7.QUANTO_8BIT,
            Models.Opt.HuggingFace.B6_7.QUANTO_4BIT,
            Models.Opt.HuggingFace.B6_7.QUANTO_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-6.7b")
        for model_config in [
            Models.Opt.HuggingFace.B13.QUANTO_8BIT,
            Models.Opt.HuggingFace.B13.QUANTO_4BIT,
            Models.Opt.HuggingFace.B13.QUANTO_2BIT,
        ]:
            self._register_model(model_config, "facebook/opt-13b")
    
    def _register_model(self, identifier: ModelIdentifier, model_path: str, tokenizer_path: str = None):
        """Register a model with its paths"""
        if identifier.is_local:
            model_path = os.path.join(self.model_save_path, model_path)
        
        tokenizer_path = tokenizer_path or model_path
        self._model_paths[identifier] = ModelPaths(model_path, tokenizer_path)

    def get_model_paths(self, identifier: ModelIdentifier) -> ModelPaths:
        """Get model paths for a given identifier"""
        return self._model_paths.get(identifier)

    def get_all_models(self) -> Dict[str, ModelPaths]:
        """Get all registered models with their paths"""
        return {str(identifier): paths for identifier, paths in self._model_paths.items()}

    def get_model_bit_precision(self, identifier: ModelIdentifier) -> int:
        """Get bit precision for a given model"""
        if identifier.bits:
            return identifier.bits.value
        return BitPrecision.FP16.value  # Default to FP16 for base models