import numpy  
import importlib
import os
import socket
import sys
from ipdb import iex

project_path = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
sys.path.append(project_path)
sys.path.append(project_path + '/datasets')
sys.path.append(project_path + '/backbones')
sys.path.append(project_path + '/models')

sys.path.append(project_path + '/main')

import datetime
import uuid
from argparse import ArgumentParser

import setproctitle
import torch
from utils.args import add_management_args, add_experiment_args
from utils import create_if_not_exists
# from utils.continual_training import train as ctrain
from run.ood_eval_mdl import ood_eval
from run.laplace_train import laplace_train_old
from run.laplace_ood_eval import laplace_ood_eval
from run.laplace_ood_vis import laplace_ood_vis
from run import *

from accelerate.utils import set_seed
from accelerate import Accelerator

try:
    import wandb
except ImportError:
    wandb = None

# ======== NEW IMPORTS ========
import hydra
from omegaconf import OmegaConf, DictConfig
from dotenv import load_dotenv
from argparse import Namespace
# =============================

def lecun_fix():
    from six.moves import urllib 
    opener = urllib.request.build_opener()
    opener.addheaders = [('User-agent', 'Mozilla/5.0')]
    urllib.request.install_opener(opener)


def parse_args():
    parser = ArgumentParser(description='Bayesian LoRA', allow_abbrev=False)
    add_management_args(parser)
    add_experiment_args(parser)
    args = parser.parse_known_args()[0]

    # add model-specific arguments
    mod = importlib.import_module('modelwrappers.' + args.modelwrapper)
    get_parser = getattr(mod, 'get_parser')
    parser = get_parser() # the real parsing happens. 
    args = parser.parse_args()

    # set random seed
    if args.seed is not None:
        set_seed(args.seed)

    return args

# @iex
@hydra.main(version_base=None, config_path="../conf", config_name="config")
def main(cfg: DictConfig = None,args=None):
    lecun_fix()
    if cfg is not None:
        # Step 1: build parser with all default values (no CLI parsing, no required check)
        parser = ArgumentParser(description='Bayesian LoRA', allow_abbrev=False)
        add_management_args(parser)
        add_experiment_args(parser)

        # Load model-specific parser (may contain required args)
        mod = importlib.import_module('modelwrappers.' + cfg.modelwrapper)
        get_parser = getattr(mod, 'get_parser')
        parser = get_parser()

        # Step 2: Create a dummy Namespace with parser defaults (skip required enforcement)
        defaults = {action.dest: action.default for action in parser._actions if action.dest != 'help'}
        base_args = Namespace(**defaults)

        # Step 3: Override defaults with Hydra config
        flat_cfg = OmegaConf.to_container(cfg, resolve=True)
        for k, v in flat_cfg.items():
            setattr(base_args, k, v)

        # Step 4: Re-apply seed if provided
        if hasattr(base_args, "seed") and base_args.seed is not None:
            set_seed(base_args.seed)

        args = base_args

        print("\n[Hydra Config Loaded & Defaults Applied]")
        print(OmegaConf.to_yaml(cfg))
    else:
        # fallback: traditional argparse (no Hydra)
        if args is None:
            args = parse_args()
   

    os.putenv("MKL_SERVICE_FORCE_INTEL", "1")
    os.putenv("NPY_MKL_FORCE_INTEL", "1")
    
   
    # Add uuid, timestamp and hostname for logging
    args.conf_jobnum = str(uuid.uuid4())
    args.conf_timestamp = str(datetime.datetime.now())
    args.conf_host = socket.gethostname()

    accelerator = Accelerator()

    ood_ori_dataset = None
    if args.ood_ori_dataset is not None:
        dataset = args.dataset
        args.dataset = args.ood_ori_dataset
        ood_ori_dataset = get_dataset(args.dataset_type, accelerator, args)
        ood_ori_dataset.get_loaders()
        args.ood_ori_outdim = ood_ori_dataset.num_labels # should be careful to use in evaluate_all
        args.ood_ori_num_samples = ood_ori_dataset.num_samples
        args.dataset = dataset

    dataset = get_dataset(args.dataset_type, accelerator, args)

    dataset.get_loaders()
    args.outdim = dataset.num_labels 
    args.num_samples = dataset.num_samples

    model = get_model(args, accelerator)
    model.model.print_trainable_parameters()
    setproctitle.setproctitle('{}_{}_BLoB-lora'.format(args.model, args.dataset))

   
    
    if args.ood_ori_dataset is not None: 
        print('INSIDE OOD EVALUATION')
        print('ood eval:')
        print(ood_eval)
        print('model.model.target_ids')
        print(model.model.target_ids)
        ood_eval(model, dataset, accelerator, args, ood_ori_dataset)
    else: 
        wandb_logger = None
        if accelerator.is_local_main_process:
            print(args)
            if not args.nowand:
                assert wandb is not None, "Wandb not installed, please install it or run without wandb"
                if not args.wandb_name:
                    wandb_logger = wandb.init(project=args.wandb_project, entity=args.wandb_entity, config=vars(args), reinit="finish_previous")
                else:
                    wandb_logger = wandb.init(project=args.wandb_project, entity=args.wandb_entity, name=args.wandb_name, config=vars(args), reinit="finish_previous")
            print(file=sys.stderr)
        print('\n Prepare for fit evaluate :::::::::::::\n') 
        model.model.prepare_for_fit_evaluate(dataset, wandb_logger)
        print('\n fit evaluate ::::::::::::: \n')
        model.model.fit_evaluate()
    

        if not args.nowand:
            if accelerator.is_local_main_process:
                wandb_logger.finish()


if __name__ == '__main__':
    main()
