from transformers import LlamaModel, AutoTokenizer, AutoModel, Trainer, default_data_collator, TrainerCallback, TrainingArguments
from contextlib import nullcontext
import torch
import torch.distributed as dist
import numpy as np
from utils.args import Arguments
from utils.dist import is_dist, set_dist_env
from utils.metrics import accuracy
from utils.peft import create_lora_config
from utils.utils import model_id
from models.LMs import BertClassifier, LlamaClassifier, AttBertClassifier, TransAttBertClassifier
from data.load import load_data
from data.dataset import NCDataset, A_NCDataset
from data.sampling import collect_subgraphs, constrained_collect_subgraphs, constrained_ego_graphs_sampler
import os
from copy import deepcopy

os.environ["TOKENIZERS_PARALLELISM"] = "false"

def collect_txt(idx, txt):
    tmp = []
    for i in idx:
        tmp.append(txt[i])
    return tmp

def predict_and_concatenate(model, dataset, batch_size, device, training_args):
    num_subsets = 10
    subset_size = len(dataset) // num_subsets
    
    # 收集 NumPy 数组的列表
    predictions_zero_list = []
    label_ids_list = []
    all_token_embeddings_list = []
    all_low_embeddings_list = []
    all_boundary_list = []
    all_block_rules_list = []
    all_root_list = []
    all_sorted_sequence_list = []

    model.eval()

    for i in range(num_subsets):
        start_idx = i * subset_size
        end_idx = (i + 1) * subset_size if i != num_subsets - 1 else len(dataset)
        subset_indices = list(range(start_idx, end_idx))
        
        subset_dataset = torch.utils.data.Subset(dataset, subset_indices)
        
        trainer = Trainer(
            model=model,
            args=training_args,
            data_collator=default_data_collator,
        )

        
        
        predictions = trainer.predict(subset_dataset)
        
        # 1. 收集主预测 logits
        predictions_zero_list.append(predictions.predictions[0])  # (subset_size, num_classes)
        
        subset_labels = dataset.labels[start_idx:end_idx] # 形状 (subset_size,)
        # 转换为 numpy 以便后续 concatenate
        label_ids_list.append(subset_labels.cpu().numpy()) 
        
        # 3. 收集额外输出
        extra_outputs = predictions.predictions[1]  # 假设是 (token_emb, low_emb, boundary, block_rules, root, sorted_seq)
        token_embeddings, low_embeddings, boundary, block_rules, root, sorted_sequences = extra_outputs
        
        all_token_embeddings_list.append(token_embeddings)
        all_low_embeddings_list.append(low_embeddings)
        all_boundary_list.append(boundary)
        all_block_rules_list.append(block_rules)
        all_root_list.append(root)
        all_sorted_sequence_list.append(sorted_sequences)

        # 清理内存
        del trainer, predictions
        torch.cuda.empty_cache()

    

    # 4. 使用 np.concatenate 拼接所有子集
    # axis=0 表示在第一个维度（样本数）上拼接
    prediction_zero = np.concatenate(predictions_zero_list, axis=0)      # (total_size, num_classes)
    label_ids = np.concatenate(label_ids_list, axis=0)                   # (total_size,)
    all_token_embeddings = np.concatenate(all_token_embeddings_list, axis=0)
    all_low_embeddings = np.concatenate(all_low_embeddings_list, axis=0)
    all_boundary = np.concatenate(all_boundary_list, axis=0)
    all_block_rules = np.concatenate(all_block_rules_list, axis=0)
    all_root = np.concatenate(all_root_list, axis=0)
    all_sorted_sequence = np.concatenate(all_sorted_sequence_list, axis=0)

    return prediction_zero, label_ids, all_token_embeddings, all_low_embeddings, all_boundary, all_block_rules, all_root, all_sorted_sequence

if __name__ == '__main__':
    if is_dist():
        rank = set_dist_env()
    else:
        rank = 0
     
    config = Arguments().parse_args()
    print(config)
    
    device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')
    
    output_dir = f"tmp"
    epochs = config.epochs
    enable_profiler = False
    
    # Set up profiler
    if enable_profiler:
        wait, warmup, active, repeat = 1, 1, 2, 1
        total_steps = (wait + warmup + active) * (1 + repeat)
        schedule =  torch.profiler.schedule(wait=wait, warmup=warmup, active=active, repeat=repeat)
        profiler = torch.profiler.profile(
            schedule=schedule,
            on_trace_ready=torch.profiler.tensorboard_trace_handler(f"{output_dir}/logs/tensorboard"),
            record_shapes=True,
            profile_memory=True,
            with_stack=True)
        
        class ProfilerCallback(TrainerCallback):
            def __init__(self, profiler):
                self.profiler = profiler
                
            def on_step_end(self, *args, **kwargs):
                self.profiler.step()

        profiler_callback = ProfilerCallback(profiler)
    else:
        profiler = nullcontext()
    
    acc_list = []
    best_val_acc = 0
    
    for run in range(config.runs):
        
        data, text, num_classes = load_data(config.dataset, use_text=True, seed=run)

        # Load model from HuggingFace Hub
        if config.lm_type == 'llama':
            tokenizer = AutoTokenizer.from_pretrained(model_id[config.lm_type], use_fast=False, trust_remote_code=True)
            bert_model = LlamaModel.from_pretrained(model_id[config.lm_type], device_map=rank if is_dist() else device, torch_dtype=torch.float16)
            tokenizer.sep_token = tokenizer.bos_token
            bert_model.config.sep_token_id = bert_model.config.bos_token_id
            tokenizer.pad_token = tokenizer.eos_token
            bert_model.config.pad_token_id = bert_model.config.eos_token_id
        else:
            # bert_model = AutoModel.from_pretrained(model_id[config.lm_type], output_hidden_states=True, return_dict=True)
            # tokenizer = AutoTokenizer.from_pretrained(model_id[config.lm_type])
            model_path = f"../models/{config.lm_type}"
            tokenizer = AutoTokenizer.from_pretrained( 
                model_path,
                local_files_only=True  # 强制只使用本地文件
            )
            bert_model = AutoModel.from_pretrained( 
                model_path,
                output_hidden_states=True,
                return_dict=True,
                local_files_only=True  # 强制只使用本地文件
            ).to(f"cuda:{rank}")
        
        train_idx = data.train_mask.nonzero().squeeze()
        val_idx = data.val_mask.nonzero().squeeze()
        test_idx = data.test_mask.nonzero().squeeze() 

        graphs = constrained_ego_graphs_sampler(torch.arange(data.num_nodes), data, 40)
        #graphs = constrained_collect_subgraphs(torch.arange(data.num_nodes), data, walk_steps=config.walk_steps, restart_ratio=config.restart)
        #graphs = collect_subgraphs(torch.arange(data.num_nodes), data, walk_steps=config.walk_steps, restart_ratio=config.restart)
        # dataset = NCDataset(graphs, data.y, tokenizer, text, config)
        dataset = A_NCDataset(graphs, data.y, tokenizer, text, config)
        train_dataset = torch.utils.data.Subset(dataset, train_idx)
        val_dataset = torch.utils.data.Subset(dataset, val_idx)
        test_dataset =  torch.utils.data.Subset(dataset, test_idx)
        
        if config.lora:
            print("lora_activated")
            bert_model, _ = create_lora_config(bert_model, config.rank)
            if config.lm_type == 'llama':
                for param in bert_model.parameters():
                    if param.requires_grad:
                        param.data = param.data.float()
        
        
        if config.lm_type == 'llama':
            model = LlamaClassifier(bert_model, num_classes)
        else:
            # model = BertClassifier(bert_model, num_classes)
            model = TransAttBertClassifier(bert_model, num_classes)
        
        if is_dist():
            dist.barrier()
            
        # Define training args
        training_args = TrainingArguments(
            output_dir=output_dir,
            overwrite_output_dir=True,
            report_to="none",
            bf16=False,  # Use BF16 if available
            dataloader_pin_memory=False,
            # logging strategies
            logging_dir=f"{output_dir}/logs",
            logging_strategy="epoch",
            logging_steps=1,
            save_strategy="no",
            optim="adamw_torch_fused",
            max_steps=total_steps if enable_profiler else -1,
            learning_rate=config.lr,
            num_train_epochs=epochs,
            gradient_accumulation_steps=2,
            per_device_train_batch_size=config.batch_size,
            gradient_checkpointing=False,
            local_rank=rank if is_dist() else -1,
        )

        with profiler:
            # Create Trainer instance
            trainer = Trainer(
                model=model,
                args=training_args,
                train_dataset=train_dataset,
                eval_dataset=val_dataset,
                data_collator=default_data_collator,
                callbacks=[profiler_callback] if enable_profiler else [],
            )

            # Start training
            trainer.train()

        # evaluation
        # predictions = trainer.predict(dataset)
        prediction_zero, label_ids, all_token_embeddings, all_low_embeddings, all_boundary, all_block_rules, all_root, all_sorted_sequence = predict_and_concatenate(model, dataset, config.batch_size, device, training_args)
        train_predictions, train_labels = prediction_zero[train_idx], label_ids[train_idx]
        val_predictions, val_labels = prediction_zero[val_idx], label_ids[val_idx]
        test_predictions, test_labels = prediction_zero[test_idx], label_ids[test_idx]
        
        if not is_dist() or rank == 0:
            # report acc
            train_acc = accuracy(train_predictions, train_labels)
            test_acc = accuracy(test_predictions, test_labels)
            val_acc = accuracy(val_predictions, val_labels)
            
            print(f"# lr : {config.lr}  run : {run} , train acc : {train_acc * 100:.2f}, val acc : {val_acc * 100:.2f} , test acc : {test_acc * 100:.2f}")
            acc_list.append(test_acc)

            if val_acc > best_val_acc:
                best_val_acc = val_acc
                best_token_embeddings = deepcopy(all_token_embeddings)
                best_low_embeddings = deepcopy(all_low_embeddings)
                best_boundary = deepcopy(all_boundary)
                best_block_rules = deepcopy(all_block_rules)
                best_root = deepcopy(all_root)
                best_sorted_sequences = deepcopy(all_sorted_sequence)

        # if not is_dist() or rank == 0:
        #     out_dir = os.path.join('../models', 'lm_lora', f"{config.lm_type}")
        #     os.makedirs(out_dir, exist_ok=True)
        #     bert_model.save_pretrained(out_dir)   # 保存LoRA权重 
        #     #tokenizer.save_pretrained(out_dir,  save_only_config=True)    # 保存tokenizer 
        
        # clear cache
        # del predictions
        del trainer
        del model
        del tokenizer
        del bert_model
        
        if is_dist():
            dist.barrier()
        
            
    if not is_dist() or rank == 0:
        final_acc, final_acc_std = np.mean(acc_list), np.std(acc_list)
        print(f"final_acc: {final_acc * 100:.2f} ± {final_acc_std * 100:.2f}")
        out_dir = os.path.join('out', 'lm_token_embeddings', f"{config.lm_type}")
        os.makedirs(out_dir, exist_ok=True)
        
        torch.save(torch.tensor(best_token_embeddings), os.path.join(out_dir, f'{config.dataset}.pt'))

        out_dir = os.path.join('out', 'lm_low_embeddings', f"{config.lm_type}")
        os.makedirs(out_dir, exist_ok=True)
        
        torch.save(torch.tensor(best_low_embeddings), os.path.join(out_dir, f'{config.dataset}.pt'))

        out_dir = os.path.join('out', 'lm_boundary', f"{config.lm_type}")
        os.makedirs(out_dir, exist_ok=True)
        
        torch.save(torch.tensor(best_boundary), os.path.join(out_dir, f'{config.dataset}.pt'))

        out_dir = os.path.join('out', 'lm_root', f"{config.lm_type}")
        os.makedirs(out_dir, exist_ok=True)
        
        torch.save(torch.tensor(best_root), os.path.join(out_dir, f'{config.dataset}.pt'))

        out_dir = os.path.join('out', 'lm_block_rules', f"{config.lm_type}")
        os.makedirs(out_dir, exist_ok=True)
        
        torch.save(torch.tensor(best_block_rules), os.path.join(out_dir, f'{config.dataset}.pt'))

        out_dir = os.path.join('out', 'lm_sorted_sequences', f"{config.lm_type}")
        os.makedirs(out_dir, exist_ok=True)
        
        torch.save(torch.tensor(best_sorted_sequences), os.path.join(out_dir, f'{config.dataset}.pt'))