import torch
from torch.nn import CrossEntropyLoss
from tqdm import tqdm
# https://huggingface.co/docs/accelerate/v0.11.0/en/memory#accelerate.find_executable_batch_size
from accelerate.utils import find_executable_batch_size
from ..baseline import SingleBehaviorRedTeamingMethod
from ..model_utils import get_template
from copy import deepcopy
import json


# ============================== UTILS FOR UAT ============================== #
def hotflip_attack(averaged_grad, embedding_matrix, trigger_token_ids,
                   increase_loss=False, num_candidates=1):
    """
    The "Hotflip" attack described in Equation (2) of the paper. This code is heavily inspired by
    the nice code of Paul Michel here https://github.com/pmichel31415/translate/blob/paul/
    pytorch_translate/research/adversarial/adversaries/brute_force_adversary.py

    This function takes in the model's average_grad over a batch of examples, the model's
    token embedding matrix, and the current trigger token IDs. It returns the top token
    candidates for each position.

    If increase_loss=True, then the attack reverses the sign of the gradient and tries to increase
    the loss (decrease the model's probability of the true class). For targeted attacks, you want
    to decrease the loss of the target class (increase_loss=False).
    """
    averaged_grad = averaged_grad
    embedding_matrix = embedding_matrix
    trigger_token_embeds = torch.nn.functional.embedding(torch.LongTensor(trigger_token_ids).cuda(),
                                                         embedding_matrix).detach().unsqueeze(0)
    averaged_grad = averaged_grad.unsqueeze(0)

    # einsum does not support half precision, so we rewrite as bmm.
    gradient_dot_embedding_matrix = torch.bmm(averaged_grad, torch.transpose(embedding_matrix, 0, 1).unsqueeze(0))
    
    if not increase_loss:
        gradient_dot_embedding_matrix *= -1    # lower versus increase the class probability.
    if num_candidates > 1: # get top k options
        _, best_k_ids = torch.topk(gradient_dot_embedding_matrix, num_candidates, dim=2)
        return best_k_ids.detach().cpu().numpy()[0]
    _, best_at_each_step = gradient_dot_embedding_matrix.max(2)
    return best_at_each_step[0].detach().cpu().numpy()


# ============================== GCG CLASS DEFINITION ============================== #
class UAT(SingleBehaviorRedTeamingMethod):
    def __init__(self,
                 target_model,
                 num_steps=50,
                 adv_string_init="! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! ! !",
                 use_prefix_cache=True,
                 num_candidates=3,
                 targets_path=None,
                 eval_steps=10,
                 **model_kwargs):
        """
        :param num_steps: the number of optimization steps to use
        :param adv_string_init: the initial adversarial string to start with
        :param early_stopping: whether to stop early if the loss is less than 0.1
        """
        super().__init__(target_model, **model_kwargs)  # assigns self.model, self.tokenizer, and self.model_name_or_path
        self.num_steps = num_steps
        self.adv_string_init = adv_string_init
        self.use_prefix_cache = use_prefix_cache
        self.num_candidates = num_candidates

        if targets_path is None:
            raise ValueError("targets_path must be specified")
        with open(targets_path, 'r', encoding='utf-8') as file:
            self.behavior_id_to_target = json.load(file)
        self.targets_path = targets_path
        
        ### Eval Vars ###
        self.eval_steps = eval_steps

        template = get_template(**target_model)['prompt']
        self.template = template
        self.before_tc, self.after_tc = template.split("{instruction}")

    def extract_grad_hook(self, module, grad_in, grad_out):
        self.extracted_grads.append(grad_out[0])

    def get_loss(self, search_batch_size, input_embeds, target_embeds, target_ids, no_grad=True):
        loss_list = []
        for i in range(0, input_embeds.shape[0], search_batch_size):
            input_embeds_batch = input_embeds[i:i+search_batch_size]
            current_batch_size = input_embeds_batch.shape[0]

            # forward pass
            if self.use_prefix_cache:
                # Expand prefix cache to match batch size
                prefix_cache_batch = []
                for i in range(len(self.prefix_cache)):
                    prefix_cache_batch.append([])
                    for j in range(len(self.prefix_cache[i])):
                        prefix_cache_batch[i].append(self.prefix_cache[i][j].expand(current_batch_size, -1, -1, -1))
                if no_grad:
                    with torch.no_grad():
                        outputs = self.model(inputs_embeds=input_embeds_batch, past_key_values=prefix_cache_batch)
                else:
                    outputs = self.model(inputs_embeds=input_embeds_batch, past_key_values=prefix_cache_batch)
            else:
                if no_grad:
                    with torch.no_grad():
                        outputs = self.model(inputs_embeds=input_embeds_batch)
                else:
                    outputs = self.model(inputs_embeds=input_embeds)

            logits = outputs.logits
            # compute loss
            # Shift so that tokens < n predict n
            tmp = input_embeds.shape[1] - target_embeds.shape[1]
            shift_logits = logits[..., tmp-1:-1, :].contiguous()
            shift_labels = target_ids.repeat(current_batch_size, 1)
            # Flatten the tokens
            loss_fct = CrossEntropyLoss(reduction='none')
            loss = loss_fct(shift_logits.view(-1, shift_logits.size(-1)), shift_labels.view(-1))
            loss = loss.view(current_batch_size, -1).mean(dim=1)
            loss_list.append(loss)
        loss = torch.cat(loss_list, dim=0)
        return loss

    def generate_test_cases_single_behavior(self, behavior, verbose=False, **kwargs):
        """
        Generate a test cases for the behavior

        :param behavior: the behavior to generate test cases for
        :param verbose: whether to print progress
        """
        # starting search_batch_size, will auto reduce batch_size later if go OOM (resets for every new behavior)
        self.search_batch_size = self.num_candidates

        # unpack list from behaviors.csv row
        # ========== Behavior meta data ==========
        behavior_dict = behavior
        behavior = behavior_dict['Behavior']
        context_str = behavior_dict['ContextString']
        behavior_id = behavior_dict['BehaviorID']
        target = self.behavior_id_to_target[behavior_id]
        behavior += ' '
        ### Targeted Model and Tokenier ###
        model = self.model
        device = model.device
        tokenizer = self.tokenizer

        ### vars ###
        adv_string_init = self.adv_string_init
        num_steps = self.num_steps
        before_tc = self.before_tc
        after_tc = self.after_tc
        eval_steps = self.eval_steps

        # handle setup for behaviors with context strings
        if context_str:
            behavior = f"{context_str}\n\n---\n\n{behavior}"

        print(f'Behavior: {behavior_id} || Target: {target}')

        # ========= add gradient hooks to embeddings =========
        embed_layer = model.get_input_embeddings()
        embed_layer.weight.requires_grad = True
        embed_layer.register_backward_hook(self.extract_grad_hook)
        embedding_weight = embed_layer.weight.detach()


        # ========== setup trigger_tokens and input_embeds components ========== #
        trigger_tokens = tokenizer(adv_string_init, return_tensors="pt", add_special_tokens=False).to(device)['input_ids'][0]
        trigger_token_length = len(trigger_tokens)

        behavior_ids = tokenizer(behavior, return_tensors="pt", add_special_tokens=False).to(device)['input_ids']
        behavior_embeds = embed_layer(behavior_ids)
        target_ids = tokenizer(target, return_tensors="pt", add_special_tokens=False).to(device)['input_ids']
        target_embeds = embed_layer(target_ids)

        before_ids = tokenizer(before_tc, return_tensors="pt").to(device)['input_ids']
        before_embeds = embed_layer(before_ids)
        after_ids = tokenizer(after_tc, return_tensors="pt", add_special_tokens=False).to(device)['input_ids']
        after_embeds = embed_layer(after_ids)

        if self.use_prefix_cache:
            # precompute and cache states for system prompt and user prompt
            init_input_embeds = torch.cat([before_embeds, behavior_embeds], dim=1)
            with torch.no_grad():
                outputs = model(inputs_embeds=init_input_embeds, use_cache=True)
                self.prefix_cache = outputs.past_key_values
        
        # ========== get initial loss for the trigger ==========
        model.zero_grad()
        if self.use_prefix_cache:
            input_embeds = torch.cat([embed_layer(trigger_tokens.unsqueeze(0)), after_embeds, target_embeds], dim=1)
        else:
            input_embeds = torch.cat([before_embeds, behavior_embeds, embed_layer(trigger_tokens.unsqueeze(0)), after_embeds, target_embeds], dim=1)
        loss = self.get_loss(1, input_embeds, target_embeds, target_ids, no_grad=False)
        best_loss = loss
        counter = 0
        end_iter = False

        # ========== run optimization ========== #
        all_losses = []
        all_test_cases = []
        for i in tqdm(range(num_steps), desc=f"UAT"):
            if end_iter: break

            for token_to_flip in range(0, trigger_token_length):
                if end_iter: break
                # Get average gradient w.r.t. the triggers
                extracted_grads = [] # clear the gradient from past iterations
                self.extracted_grads = []
                loss.backward(retain_graph=True)

                averaged_grad = torch.sum(self.extracted_grads[0], dim=0)
                averaged_grad = averaged_grad[token_to_flip].unsqueeze(0)

                # Use hotflip (linear approximation) attack to get the top num_candidates
                candidates = hotflip_attack(averaged_grad, embedding_weight,
                                            [trigger_tokens[token_to_flip]],
                                            increase_loss=False, num_candidates=self.num_candidates)[0]
                
                # Prepare a batch of candidate embeddings
                candidates_tokens = []
                for cand in candidates:
                    candidate_trigger_tokens = deepcopy(trigger_tokens)
                    candidate_trigger_tokens[token_to_flip] = cand
                    candidates_tokens.append(candidate_trigger_tokens)
                candidates_tokens = torch.stack(candidates_tokens, dim=0)

                candidates_embeds = embed_layer(candidates_tokens)
                if self.use_prefix_cache:
                    input_embeds_candidates = torch.cat([candidates_embeds,
                                                         after_embeds.repeat(len(candidates), 1, 1),
                                                         target_embeds.repeat(len(candidates), 1, 1)], dim=1)
                else:
                    input_embeds_candidates = torch.cat([before_embeds.repeat(len(candidates), 1, 1),
                                                         behavior_embeds.repeat(len(candidates), 1, 1),
                                                         candidates_embeds,
                                                         after_embeds.repeat(len(candidates), 1, 1),
                                                         target_embeds.repeat(len(candidates), 1, 1)], dim=1)

                # Auto Find Batch Size for foward candidates (each time go OOM will decay search_batch_size // 2)
                candidate_losses = find_executable_batch_size(self.get_loss, self.search_batch_size)(input_embeds_candidates, target_embeds, target_ids)

                curr_best_loss = candidate_losses[candidate_losses.argmin()]
                curr_best_trigger_tokens = candidates_tokens[candidate_losses.argmin()]

                # Update overall best if the best current candidate is better
                if curr_best_loss < best_loss:
                    counter = 0 # used to exit early if no improvements in the trigger
                    best_loss = curr_best_loss
                    trigger_tokens = curr_best_trigger_tokens
                # if you have gone through all trigger_tokens without improvement, end iteration
                elif counter == len(trigger_tokens):
                    end_iter = True
                # If the loss didn't get better, just move to the next word.
                else:
                    counter = counter + 1
                # reevaluate the best candidate so you can backprop into it at next iteration
                model.zero_grad()

                if self.use_prefix_cache:
                    input_embeds = torch.cat([embed_layer(trigger_tokens.unsqueeze(0)), after_embeds, target_embeds], dim=1)
                else:
                    input_embeds = torch.cat([before_embeds, behavior_embeds, embed_layer(trigger_tokens.unsqueeze(0)), after_embeds, target_embeds], dim=1)
                loss = self.get_loss(1, input_embeds, target_embeds, target_ids, no_grad=False)

            test_case_ids = torch.cat([behavior_ids, trigger_tokens.unsqueeze(0)], dim=1)
            test_case = tokenizer.decode(test_case_ids[0])

            all_losses.append(curr_best_loss.item())
            all_test_cases.append(test_case)
            # ========== Eval ========== #
            if i % eval_steps == 0:
                print('Step {} | Test Case: {} | Loss: {:.3f}'.format(i, test_case, best_loss.item()))
                print()

        logs = {'best_loss': best_loss.item(), 'all_losses': all_losses, 'all_test_cases': all_test_cases}
        return [test_case], [logs]