# Copyright (c) Alibaba, Inc. and its affiliates.
import inspect
import random
from collections import defaultdict
from contextlib import nullcontext
from typing import Any, Optional, Union

import torch
import torch.nn as nn
from transformers import PreTrainedModel
from trl import GKDTrainer as HFGKDTrainer
from trl import SFTTrainer as HFSFTTrainer
from trl.models.utils import prepare_deepspeed

from swift.utils import unwrap_model_for_generation
from ..mixin import SwiftMixin
from .rlhf_mixin import RLHFTrainerMixin

del HFGKDTrainer.__init__
del HFSFTTrainer.__init__


class GKDTrainer(RLHFTrainerMixin, SwiftMixin, HFGKDTrainer):

    def __init__(
        self,
        model: Optional[Union[PreTrainedModel, nn.Module, str]] = None,
        *_args,
        **kwargs
    ):
        teacher_model = kwargs.pop("teacher_model")
        super().__init__(model, *_args, **kwargs)
        args = kwargs["args"]
        self.lmbda = args.lmbda
        self.temperature = args.temperature
        self.seq_kd = args.seq_kd
        self.generation_config = model.generation_config
        self._metrics = {"train": defaultdict(list), "eval": defaultdict(list)}
        self._total_train_tokens = 0
        if self.is_deepspeed_enabled:
            self.teacher_model = prepare_deepspeed(teacher_model, self.accelerator)
        else:
            self.teacher_model = self.accelerator.prepare_model(
                teacher_model, evaluation_mode=True
            )
        self.teacher_model.eval()
        # Initialize activation offloading context
        args.activation_offloading = False  # TODO: remove
        if args.activation_offloading:
            from trl.models import get_act_offloading_ctx_manager

            self.maybe_activation_offload_context = get_act_offloading_ctx_manager(
                model=self.model
            )
        else:
            self.maybe_activation_offload_context = nullcontext()

    # Code borrowed from huggingface/trl
    def generate_on_policy_outputs(
        self, model, inputs, generation_config, pad_token_id=None
    ):
        assert not self.template._packing, "generate not support padding_free/packing."
        # Generate output with respect to the prompt only
        model_inputs = {
            k: v
            for k, v in inputs.items()
            if not k.startswith("prompt") and k != "labels"
        }
        model_inputs["input_ids"] = inputs["prompts"]
        model_inputs.update(
            {
                k[len("prompt_") :]: v
                for k, v in inputs.items()
                if k.startswith("prompt_")
            }
        )
        model_inputs.pop("position_ids", None)
        kwargs = {}
        base_model = self.template.get_base_model(model)
        parameters = inspect.signature(base_model.generate).parameters
        if "use_model_defaults" in parameters:
            kwargs["use_model_defaults"] = False
        with self.template.generate_context():
            if self.model.model_meta.is_multimodal:
                _, model_inputs = self.template.pre_forward_hook(
                    model, None, model_inputs
                )
            generated_outputs = model.generate(
                **model_inputs,
                generation_config=generation_config,
                return_dict_in_generate=True,
                **kwargs
            )
        # Get the generated token IDs
        generated_tokens = generated_outputs.sequences
        if not self.template.skip_prompt:
            generated_tokens = torch.concat(
                [inputs["prompts"], generated_tokens], dim=1
            )
        # Calculate new attention mask
        new_attention_mask = torch.ones_like(generated_tokens)
        new_labels = generated_tokens.clone()
        new_labels[:, : inputs["prompts"].shape[1]] = -100

        # If there's pad_token_id, set attention mask to 0 for padding tokens
        if pad_token_id is not None:
            new_labels[new_labels == pad_token_id] = -100
            new_attention_mask[generated_tokens == pad_token_id] = 0

        new_position_ids = new_attention_mask.cumsum(dim=1) - 1
        new_position_ids[new_position_ids < 0] = 0
        inputs["position_ids"] = new_position_ids
        return generated_tokens, new_attention_mask, new_labels

    def compute_loss(
        self, model, inputs, return_outputs=False, num_items_in_batch=None
    ):
        model_inputs = {
            k: v for k, v in inputs.items() if k not in {"prompt", "labels"}
        }
        # If generate is used, then use_logits_to_keep must be set to False.
        use_logits_to_keep = self.get_use_logits_to_keep(True)
        if use_logits_to_keep:
            inputs["labels"], logits_to_keep = self.get_logits_to_keep(inputs["labels"])
            if logits_to_keep is not None:
                model_inputs["logits_to_keep"] = logits_to_keep
        if self.args.sft_alpha > 0:
            model_inputs["labels"] = inputs["labels"]
        # compute student output
        outputs_student = model(**model_inputs)

        model_inputs.pop("labels", None)
        with torch.no_grad():
            outputs_teacher = self.teacher_model(**model_inputs)

        shifted_labels = torch.roll(inputs["labels"], shifts=-1, dims=1)
        mask = shifted_labels != -100
        shifted_student_logits = outputs_student.logits[mask][None]
        shifted_teacher_logits = outputs_teacher.logits[mask][None]

        # compute loss
        loss = self.generalized_jsd_loss(
            student_logits=shifted_student_logits,
            teacher_logits=shifted_teacher_logits,
            beta=self.beta,
        )
        if self.args.sft_alpha > 0:
            loss = loss + self.args.sft_alpha * outputs_student.loss

        # Return loss
        return (loss, outputs_student) if return_outputs else loss

    # Code borrowed from huggingface/trl
    def training_step(
        self,
        model: nn.Module,
        inputs: dict[str, Union[torch.Tensor, Any]],
        num_items_in_batch: Optional[int] = None,
    ) -> torch.Tensor:
        """
        Perform a training step for the Generalized Knowledge Distillation (GKD) model.

        This method implements the on-policy learning approach described in the GKD paper.
        With probability `self.lmbda`, it generates new responses using the student model,
        which are then used for training instead of the original inputs.
        """

        if random.random() <= self.lmbda:
            with unwrap_model_for_generation(
                model,
                self.accelerator,
                gather_deepspeed3_params=self.args.ds3_gather_for_generation,
            ) as unwrapped_model:
                new_input_ids, new_attention_mask, new_labels = (
                    self.generate_on_policy_outputs(
                        unwrapped_model,
                        inputs,
                        self.generation_config,
                        self.processing_class.pad_token_id,
                    )
                )
            inputs["input_ids"] = new_input_ids
            inputs["attention_mask"] = new_attention_mask
            inputs["labels"] = new_labels
        elif self.seq_kd:
            with unwrap_model_for_generation(
                self.teacher_model,
                self.accelerator,
                gather_deepspeed3_params=self.args.ds3_gather_for_generation,
            ) as unwrapped_model:
                new_input_ids, new_attention_mask, new_labels = (
                    self.generate_on_policy_outputs(
                        unwrapped_model,
                        inputs,
                        self.generation_config,
                        self.processing_class.pad_token_id,
                    )
                )
            inputs["input_ids"] = new_input_ids
            inputs["attention_mask"] = new_attention_mask
            inputs["labels"] = new_labels

        with self.template.forward_context(self.model, inputs):
            loss = HFSFTTrainer.training_step(self, model, inputs, num_items_in_batch)
        return loss

    def prediction_step(self, model, inputs, *args, **kwargs):
        with self.template.forward_context(self.model, inputs):
            return super().prediction_step(model, inputs, *args, **kwargs)
