import torch
from torch.utils.data import DataLoader
from transformers import (
    RobertaTokenizer,
    RobertaForSequenceClassification,
    AdamW,
    get_linear_schedule_with_warmup,
    TrainingArguments,
    Trainer
)
from datasets import load_dataset
from tqdm.auto import tqdm
import numpy as np
from peft import get_peft_model, LoraConfig, TaskType
import argparse
import warnings
import os
from datetime import datetime
import json
import yaml
import atexit
import wandb

from fed_sb.utils.data_utils import *
from fed_sb.models import *
from fed_sb.utils.initialization_utils import *
from fed_sb.utils.gradient_utils import *
from fed_sb.utils.misc import *
from fed_sb.utils.merge_adapter_to_base_model import *

from fed_sb.fed.fed_data_utils import *
from fed_sb.fed.fed_agg import *
from fed_sb.fed.fed_utils import *
from fed_sb.train_eval import *
from fed_sb.utils.data_utils import *
from fed_sb.models import *
from fed_sb.utils.initialization_utils import *
from fed_sb.utils.gradient_utils import *
from fed_sb.utils.misc import *

import os
os.environ['MASTER_ADDR'] = 'localhost'
os.environ['MASTER_PORT'] = '12355'

def create_run_directory(args):
    """Create a directory structure for the current training run."""
    # Create base directory for all runs
    base_dir = "experiments/instruction_tuning"
    
    # Create timestamp for unique run identification
    timestamp = datetime.now().strftime("%Y%m%d_%H%M%S")
    
    # Create model name directory (simplified name)
    model_name = args.model.split('/')[-1]
    
    # Create run-specific directory with relevant parameters
    run_name = f"{model_name}__r{args.lora_r}__lr{args.lr}__train_{args.dataset_split.replace('[:','').replace(']','')}"
    
    # Final directory structure: experiments/model_name/YYYYMMDD_HHMMSS_parameters
    run_dir = os.path.join(base_dir, model_name, args.agg_type, f"{timestamp}_{run_name}")
    
    # Create directories
    os.makedirs(run_dir, exist_ok=True)
    os.makedirs(os.path.join(run_dir, "checkpoints"), exist_ok=True)
    os.makedirs(os.path.join(run_dir, "logs"), exist_ok=True)
    
    # Save run configuration
    config_dict = vars(args)
    with open(os.path.join(run_dir, "config.json"), 'w') as f:
        json.dump(config_dict, f, indent=4)
    
    return run_dir

def finetune():
    run_dir = create_run_directory(args)
    
    # Initialize wandb with the run directory
    wandb_run_name = os.path.basename(run_dir)
    wandb_run = wandb.init(
        project="fed-sb-arithmetic-2",
        config=args,
        dir=os.path.join(run_dir, "logs")
    )

    # Save wandb run ID to a file
    with open(os.path.join(run_dir, "wandb_run_id.txt"), "w") as f:
        f.write(wandb_run.id)

    
    
    # Create model and tokenizer
    model, tokenizer = create_model_tokenizer_it(args)
    
    # Data handling
    train_dataset = load_and_preprocess_it(tokenizer=tokenizer, args=args)

    data_collator = DataCollatorForSupervisedDataset(tokenizer=tokenizer)

    if args.agg_type == "fed-sb":
        train_loader = DataLoader(
            train_dataset, 
            batch_size=args.eg_bs, 
            shuffle=True, 
            collate_fn=data_collator
        )
        
        named_grads = None

        total_training_steps = len(train_loader) * args.epochs

        eff_lr = args.lr/(args.warmup_ratio * total_training_steps)


        named_grads = estimate_and_process_grads_torch(
            model=model,
            dataloader=train_loader,
            lr=eff_lr,
            num_samples=args.num_samples,
        )

        # Create peft model
        if args.agg_type == "ffa":
            model, lora_config = create_peft_FFA_model_it(model, args)
        else:
            model, lora_config = create_peft_model_it(model, args)

            
        # Convert model to xs

        reconstr_config_path = os.path.join(run_dir, "reconstruct_config.yaml")
        # Copy reconstruct config to run directory
        with open("config/reconstruct_config.yaml", 'r') as src, open(reconstr_config_path, 'w') as dst:
            reconstr_config = yaml.load(src, Loader=yaml.FullLoader)
            reconstr_config['svd']['rank'] = args.lora_r
            yaml.dump(reconstr_config, dst)

        # Save the required JSON file with the correct name
        json_path = os.path.join(run_dir, "reconstr_config.json")  # Note: reconstr not reconstruct
        with open(json_path, 'w') as f:
            json.dump(reconstr_config, f, indent=4)
        
        adapter_name = "default"
        peft_config_dict = {adapter_name: lora_config}

        

        named_grads_new = {f'base_model.model.{k}': v for k, v in named_grads.items()}

        del model
        model, tokenizer = create_model_tokenizer_it(args)
        
        
        if named_grads is not None:
            del named_grads
        for i in range(args.num_clients):
            # Create client dataset
            client_dataset = train_dataset.select(range(i*len(train_dataset)//args.num_clients, 
                                                    (i+1)*len(train_dataset)//args.num_clients))

            data_module = dict(train_dataset=client_dataset, data_collator=data_collator)

            
            if args.agg_type == "ffa":
                client_model, lora_config = create_peft_FFA_model_it(model, args)
            else:
                client_model,lora_config = create_peft_model_it(model, args)
            
            adapter_name = "default"
            
            peft_config_dict = {adapter_name: lora_config}
            find_and_initialize_grad(
                model=client_model,
                peft_config=peft_config_dict,
                adapter_name=adapter_name,
                reconstr_type='svd',
                reconstruct_config=reconstr_config,
                writer=None,
                named_grads=named_grads_new,
            )
            for param in client_model.parameters():
                param.data = param.data.contiguous()
            optimizer = AdamW(client_model.parameters(), lr=args.lr)


            # Training arguments
            training_args = TrainingArguments(
                output_dir=os.path.join(run_dir, "checkpoints"),
                num_train_epochs=args.epochs,
                per_device_train_batch_size=args.batch_size,
                learning_rate=args.lr,
                weight_decay=0,
                warmup_ratio=args.warmup_ratio,
                lr_scheduler_type=args.scheduler,
                seed=args.seed,
                report_to="wandb",
                gradient_accumulation_steps=32,
                save_strategy="no",
                bf16=True,
                tf32=False,
                fp16=False,
                logging_steps=1,
                logging_first_step=True,
                logging_dir=os.path.join(run_dir, "logs")
            )

            # Save training arguments
            training_args_path = os.path.join(run_dir, "training_args.json")
            with open(training_args_path, 'w') as f:
                json.dump(training_args.to_dict(), f, indent=4)

            # Create trainers
            trainer = Trainer(
                model=client_model,
                args=training_args,
                **data_module,
                optimizers=(optimizer, None),
            )
        
            # Save tokenizer
            tokenizer.save_pretrained(os.path.join(run_dir, "tokenizer"))

            client_model.config.use_cache = False
            trainer.train()

            final_model_path = os.path.join(run_dir, f"final_model_{i}")  # Fixed path naming
            trainer.save_state()
            client_model.save_pretrained(final_model_path)
            print(f"Saved model {i} to {final_model_path}")

        return run_dir

    # Split datasets and create models for each client
    else:
        for i in range(args.num_clients):
            # Create client dataset
            client_dataset = train_dataset.select(range(i*len(train_dataset)//args.num_clients, 
                                                    (i+1)*len(train_dataset)//args.num_clients))

            data_module = dict(train_dataset=client_dataset, data_collator=data_collator)

            # Create client model and optimizer
            if args.agg_type == "ffa":
                client_model, lora_config = create_peft_FFA_model_it(model, args)
            else:   
                client_model, lora_config = create_peft_model_it(model, args)
        
            optimizer = AdamW(client_model.parameters(), lr=args.lr)


            # Training arguments
            training_args = TrainingArguments(
                output_dir=os.path.join(run_dir, "checkpoints"),
                num_train_epochs=args.epochs,
                per_device_train_batch_size=args.batch_size,
                learning_rate=args.lr,
                weight_decay=0,
                warmup_ratio=args.warmup_ratio,
                lr_scheduler_type=args.scheduler,
                seed=args.seed,
                report_to="wandb",
                gradient_accumulation_steps=32,
                save_strategy="no",
                bf16=True,
                tf32=False,
                fp16=False,
                logging_steps=1,
                logging_first_step=True,
                logging_dir=os.path.join(run_dir, "logs")
            )

            # Save training arguments
            training_args_path = os.path.join(run_dir, "training_args.json")
            with open(training_args_path, 'w') as f:
                json.dump(training_args.to_dict(), f, indent=4)

            # Create trainers
            trainer = Trainer(
                model=model,
                args=training_args,
                **data_module,
                optimizers=(optimizer, None),
            )
        
            # Save tokenizer
            tokenizer.save_pretrained(os.path.join(run_dir, "tokenizer"))

            client_model.config.use_cache = False
            trainer.train()

            final_model_path = os.path.join(run_dir, f"final_model_{i}")  # Fixed path naming
            trainer.save_state()
            client_model.save_pretrained(final_model_path)
            print(f"Saved model {i} to {final_model_path}")

        return run_dir

if __name__ == "__main__":
    
    parser = argparse.ArgumentParser(description="LoRA SB for arithmetic reasoning tasks")
    
    # Dataset arguments
    parser.add_argument("--data_path", type=str, default="meta-math/MetaMathQA", help="Path to the training data")
    parser.add_argument("--dataset_split", type=str, default="train[:20000]", help="Dataset split to use. Options: ['train', 'test', 'eval']")
    parser.add_argument("--dataset_field", type=str, nargs="+", default=["query", "response"], help="Fields of dataset input and output")
    parser.add_argument("--model", type=str, default="mistralai/Mistral-7B-v0.1", help="Model name")
    parser.add_argument("--lora_r", type=int, default=200, help="LoRA R value")
    parser.add_argument("--lora_alpha", type=int, default=16, help="LoRA alpha value")
    parser.add_argument("--lora_dropout", type=float, default=0, help="LoRA dropout value")
    parser.add_argument("--batch_size", type=int, default=1, help="Batch size")
    parser.add_argument("--eg_bs", type=int, default=3, help="Batch size for gradient estimation")
    parser.add_argument("--epochs", type=int, default=1, help="Number of epochs")
    parser.add_argument("--scheduler", type=str, default="cosine", help="Learning rate scheduler")
    parser.add_argument("--warmup_ratio", type=float, default=0.02, help="Warmup ratio")
    parser.add_argument("--max_seq_length", type=int, default=512, help="Maximum sequence length")
    parser.add_argument("--lr", type=float, default=1e-4, help="Learning rate")
    parser.add_argument("--seed", type=int, default=42, help="Random seed")
    parser.add_argument("--device", type=str, default="cuda", help="Device (cuda/cpu)")
    parser.add_argument("--num_samples", type=int, default=50, help="Number of samples for gradient estimation")
    parser.add_argument("--agg_type", type=str, default="fed-sb", help="Aggregation type")
    parser.add_argument("--num_clients", type=int, default=25, help="Number of clients")
    parser.add_argument("--rounds", type=int, default=1, help="Number of rounds")
    
    args = parser.parse_args()

    # Set random seeds
    np.random.seed(args.seed)
    torch.manual_seed(args.seed)
    torch.cuda.manual_seed_all(args.seed)
    
    if args.agg_type == "fed-sb":
        args.lora_alpha = args.lora_r
    
    # Run training
    run_dir = finetune()