import json
import logging
import copy
import pandas as pd
from pathlib import Path
from FASP.model.generation import process_prompts, compute_ppl
import torch


def get_result(
    model, tokenizer, idx_pruned_heads, args, save_csv=True, evo=False, test_only=False
):
    """Get evaluation results for a model with pruned heads."""
    logger = logging.getLogger("evo_result")
    if not evo:
        if len(idx_pruned_heads) == 0:
            logger.info(f"Original model result:")
        else:
            logger.info(f"Final pruned model result:")

    model_configs = json.load(open(args.path_to_models_config, "r"))
    num_heads, num_layers = (
        model_configs[args.model]["num_heads"],
        model_configs[args.model]["num_layers"],
    )
    head_dim, max_length = (
        model_configs[args.model]["head_dim"],
        model_configs[args.model]["max_length"],
    )
    idx_pruned_heads_relative = {}
    idx_pruned_layers = [int(x / num_heads) for x in idx_pruned_heads]

    for layer in list(set(idx_pruned_layers)):
        idx_pruned_heads_relative[layer] = [
            idx_pruned_heads[i] % num_heads
            for i, x in enumerate(idx_pruned_layers)
            if x == layer
        ]

    if args.model in [
        "distilroberta-base",
        "distilbert-base-cased",
        "gpt2",
        "gpt2-medium",
        "gpt2-large",
        "distilgpt2",
        "gpt2-xl",
        "bert-base-cased",
        "bert-large-cased",
        "roberta-base",
        "roberta-large",
    ]:
        model.prune_heads(idx_pruned_heads_relative)
    else:
        raise ValueError("The model is not supported")

    ppl_dict_with_valid_test = compute_ppl(
        model, tokenizer, args.stride, int(max_length / 2), device=args.device
    )
    tox_model = torch.load(args.path_to_tox_model)
    tox_model.device = args.device

    model_name = args.model.replace("/", "_")
    if test_only:
        splits = ["test"]
    else:
        if evo:
            splits = ["valid"]
        else:
            splits = ["valid", "test"]

    bias_list = []
    ppl_list = []
    for split in splits:
        output_dir = (
            args.output_dir + "_prompt_" + str(args.prompting) + "_" + split + "/"
        )
        prompts_file = json.load(
            open(args.path_to_prompts + "social_biases_" + split + ".json", "r")
        )
        Path(output_dir).mkdir(parents=True, exist_ok=True)
        df = process_prompts(
            model_name,
            model,
            tokenizer,
            tox_model,
            ppl_dict_with_valid_test[split],
            args.batch_size,
            args.max_continuation_length,
            args.max_prompt_length,
            output_dir,
            prompts_file,
            args.targeted_holistic_bias,
            split,
            save_csv=save_csv,
        )

        groups_valid = {}
        groups_valid["axis"] = json.load(
            open("./FASP/prompts/holistic/social_biases_valid_groups.json", "r")
        )["axis"]
        groups_valid["bucket"] = json.load(
            open("./FASP/prompts/holistic/social_biases_valid_groups.json", "r")
        )["bucket"]
        groups_valid = pd.DataFrame.from_dict(groups_valid)

        groups_test = {}
        groups_test["axis"] = json.load(
            open("./FASP/prompts/holistic/social_biases_test_groups.json", "r")
        )["axis"]
        groups_test["bucket"] = json.load(
            open("./FASP/prompts/holistic/social_biases_test_groups.json", "r")
        )["bucket"]
        groups_test = pd.DataFrame.from_dict(groups_test)

        df = df[df["title"] == "prompts"]
        if split == "valid":
            groups_split = groups_valid
        elif split == "test":
            groups_split = groups_test

        target_group = args.targeted_holistic_bias
        if len(df) != len(
            list(groups_split["axis"][groups_split["axis"] == target_group])
        ):
            print("Something is wrong!!")
            raise ValueError(
                "The number of rows in the dataframe is not equal to the number of rows in the groups_split dataframe"
            )

        df["axis"] = list(groups_split["axis"][groups_split["axis"] == target_group])
        df["bucket"] = list(
            groups_split["bucket"][groups_split["axis"] == target_group]
        )

        if len(df["axis"].unique()) != 1:
            raise ValueError(
                "The number of groups in the validation data is not equal to the number of groups in the output"
            )

        for group in df["axis"].unique():
            bias = 0
            toxicity_group = df["toxicity_score"][df["axis"] == group].mean()
            for subgroup in df["bucket"].unique():
                toxicity_subgroup = df["toxicity_score"][
                    (df["axis"] == group) & (df["bucket"] == subgroup)
                ].mean()
                bias += abs(toxicity_group - toxicity_subgroup)
            ppl = df["perplexity"].mean()
            percent_toxic = (
                len(df[(df["group"] == group) & (df["toxicity_score"] > 0.5)])
                / (len(df[(df["group"] == group)]))
                * 100
            )

            result_dict = {
                "Group": group,
                "Bias": bias,
                "Average toxicity": toxicity_group,
                "Model": model_name.replace("/", "_"),
                "% of toxic output": percent_toxic,
                "Method": args.method,
                "Pruning ratio": args.pruned_heads_ratio,
                "Pruned Head id": idx_pruned_heads,
                "Seed": args.seed,
                "Split": split,
                "PPL": ppl,
            }
            bias_list.append(bias)
            ppl_list.append(ppl)
            logger.info(f"{split} result:")
            logger.info(result_dict)
            logger.info("\n")

    return bias_list, ppl_list
