"""
GPT model:
- the initial stem consists of a combination of token encoding and a positional encoding
- the meat of it is a uniform sequence of Transformer blocks
    - each Transformer is a sequential combination of a 1-hidden-layer MLP block and a self-attention block
    - all blocks feed into a central residual pathway similar to resnets
- the final decoder is a linear projection into a vanilla Softmax classifier
"""

import logging
import math
from typing import Tuple, Optional, Dict, List, Any, Union

import torch
import torch.nn as nn
from diffloss import DiffLoss
from torch.nn import functional as F

logger = logging.getLogger(__name__)

class MLPHead(nn.Module):
    """ Multi-layer perceptron head """
    def __init__(self, input_dim: int, hidden_dim: int, output_dim: int, num_layers: int = 2, dropout: float = 0.1):
        super().__init__()

        layers = []

        # Input layer
        layers.extend([
            nn.Linear(input_dim, hidden_dim),
            nn.GELU(),
            nn.Dropout(dropout)
        ])

        # Hidden layers
        for _ in range(num_layers - 2):
            layers.extend([
                nn.Linear(hidden_dim, hidden_dim),
                nn.GELU(),
                nn.Dropout(dropout)
            ])
        
        # Output layer 
        layers.append(nn.Linear(hidden_dim, output_dim))

        self.model = nn.Sequential(*layers)
        
    def forward(self, x: torch.Tensor) -> torch.Tensor:
        return self.model(x)


class GPTConfig:
    """base GPT config, params common to all GPT versions"""

    embd_pdrop = 0.1
    resid_pdrop = 0.1
    attn_pdrop = 0.1

    def __init__(self, vocab_size, block_size, **kwargs):
        self.vocab_size = vocab_size
        self.block_size = block_size
        for k, v in kwargs.items():
            setattr(self, k, v)


class GPT1Config(GPTConfig):
    """GPT-1 like network roughly 125M params"""

    n_layer = 12
    n_head = 12
    n_embd = 768
    input_dim = 12  # x, y, w, h, categories(one hot, 8)
    disc_dim = 8  # categories(one hot, 8)

    # DiffLoss config
    diffloss_d = 3
    diffloss_w = 256
    num_sampling_steps = "100"
    grad_checkpointing = False
    diffusion_batch_mul = 4


class CausalSelfAttention(nn.Module):
    """
    A vanilla multi-head masked self-attention layer with a projection at the end.
    It is possible to use torch.nn.MultiheadAttention here but I am including an
    explicit implementation here to show that there is nothing too scary here.
    """

    def __init__(self, config: GPTConfig):
        super().__init__()
        assert config.n_embd % config.n_head == 0, "Embedding dimension must be divisible by number of heads"

        # Key, query, value projections for all heads
        self.key = nn.Linear(config.n_embd, config.n_embd)
        self.query = nn.Linear(config.n_embd, config.n_embd)
        self.value = nn.Linear(config.n_embd, config.n_embd)

        # Regularization
        self.attn_drop = nn.Dropout(config.attn_pdrop)
        self.resid_drop = nn.Dropout(config.resid_pdrop)

        # Output projection
        self.proj = nn.Linear(config.n_embd, config.n_embd)

        # Causal mask to ensure attention only applies to the left in the input sequence
        mask = torch.tril(torch.ones(config.block_size, config.block_size))
        self.register_buffer(
            "mask",
            mask.view(1, 1, config.block_size, config.block_size)
        )
        self.n_head = config.n_head
        self.head_dim = config.n_embd // config.n_head

    def forward(self, x: torch.Tensor) -> torch.Tensor:
        B, T, C = x.size()

        # calculate query, key, values for all heads in batch and move head forward to be the batch dim
        k = (
            self.key(x).view(B, T, self.n_head, C // self.n_head).transpose(1, 2)
        )  # (B, nh, T, hs)
        q = (
            self.query(x).view(B, T, self.n_head, C // self.n_head).transpose(1, 2)
        )  # (B, nh, T, hs)
        v = (
            self.value(x).view(B, T, self.n_head, C // self.n_head).transpose(1, 2)
        )  # (B, nh, T, hs)

        # causal self-attention; Self-attend: (B, nh, T, hs) x (B, nh, hs, T) -> (B, nh, T, T)
        att = (q @ k.transpose(-2, -1)) * (1.0 / math.sqrt(k.size(-1)))
        att = att.masked_fill(self.mask[:, :, :T, :T] == 0, float("-inf"))
        att = F.softmax(att, dim=-1)
        att = self.attn_drop(att)
        y = att @ v  # (B, nh, T, T) x (B, nh, T, hs) -> (B, nh, T, hs)
        y = (
            y.transpose(1, 2).contiguous().view(B, T, C)
        )  # re-assemble all head outputs side by side

        # output projection
        y = self.resid_drop(self.proj(y))
        return y


class Block(nn.Module):
    """an unassuming Transformer block"""

    def __init__(self, config: GPTConfig):
        super().__init__()
        self.ln1 = nn.LayerNorm(config.n_embd)
        self.ln2 = nn.LayerNorm(config.n_embd)
        self.attn = CausalSelfAttention(config)
        self.mlp = nn.Sequential(
            nn.Linear(config.n_embd, 4 * config.n_embd),
            nn.GELU(),
            nn.Linear(4 * config.n_embd, config.n_embd),
            nn.Dropout(config.resid_pdrop),
        )

    def forward(self, x):
        x = x + self.attn(self.ln1(x))
        x = x + self.mlp(self.ln2(x))
        return x


class GPT(nn.Module):
    """the full GPT language model, with a context size of block_size"""

    def __init__(self, config: GPTConfig):
        super().__init__()

        try:
            self.diffloss = DiffLoss(
                target_channels=4,
                z_channels=config.n_embd,
                width=config.diffloss_w,
                depth=config.diffloss_d,
                num_sampling_steps=config.num_sampling_steps,
                grad_checkpointing=config.grad_checkpointing,
            )
        except:
            logger.error("DiffLoss module error.")
            self.diffloss = None

        # Input embedding stem
        self.tok_emb = nn.Linear(config.input_dim, config.n_embd)
        self.pos_emb = nn.Parameter(torch.zeros(1, config.block_size, config.n_embd))
        self.drop = nn.Dropout(config.embd_pdrop)

        # Transformer blocks
        self.blocks = nn.Sequential(*[Block(config) for _ in range(config.n_layer)])

        # Output heads
        self.ln_f = nn.LayerNorm(config.n_embd)
        self.head = MLPHead(
            input_dim = config.n_embd, 
            hidden_dim = config.n_embd * 2,
            output_dim = config.disc_dim, 
            num_layers=2, 
            dropout=0.1,
        )

        self.block_size = config.block_size
        self.diffusion_batch_mul = config.diffusion_batch_mul

        # EOS enhancement MLP
        self.eos_mlp = nn.Sequential(
            nn.Linear(config.n_embd, config.n_embd * 2),
            nn.GELU(),
            nn.Dropout(0.1),
            nn.Linear(config.n_embd * 2, 1)
        )
        self.eos_alpha = config.eos_alpha 
        self.eos_token_idx = 4      # EOS token index in discrete vocabulary
        
        self.length_loss_weight = config.length_loss_weight
        
        # Initialize weights
        self.apply(self._init_weights)

        logger.info(
            "number of parameters: %e", sum(p.numel() for p in self.parameters())
        )

    def get_block_size(self) -> int:
        return self.block_size

    def _init_weights(self, module: nn.Module) -> None:
        if isinstance(module, (nn.Linear, nn.Embedding)):
            module.weight.data.normal_(mean=0.0, std=0.02)
            if isinstance(module, nn.Linear) and module.bias is not None:
                module.bias.data.zero_()
        elif isinstance(module, nn.LayerNorm):
            if module.bias is not None:
                module.bias.data.zero_()
            if module.weight is not None:
                module.weight.data.fill_(1.0)

    def configure_optimizers(self, train_config: Any) -> torch.optim.Optimizer:
        """
        This long function is unfortunately doing something very simple and is being very defensive:
        We are separating out all parameters of the model into two buckets: those that will experience
        weight decay for regularization and those that won't (biases, and layernorm/embedding weights).
        We are then returning the PyTorch optimizer object.
        """

        # separate out all parameters to those that will and won't experience regularizing weight decay
        decay = set()
        no_decay = set()
        whitelist_weight_modules = (torch.nn.Linear,)
        blacklist_weight_modules = (torch.nn.LayerNorm, torch.nn.Embedding)
        for mn, m in self.named_modules():
            for pn, p in m.named_parameters():
                fpn = "%s.%s" % (mn, pn) if mn else pn  # full param name

                if pn.endswith("bias"):
                    # all biases will not be decayed
                    no_decay.add(fpn)
                elif pn.endswith("weight") and isinstance(m, whitelist_weight_modules):
                    # weights of whitelist modules will be weight decayed
                    decay.add(fpn)
                elif pn.endswith("weight") and isinstance(m, blacklist_weight_modules):
                    # weights of blacklist modules will NOT be weight decayed
                    no_decay.add(fpn)

        # special case the position embedding parameter in the root GPT module as not decayed
        no_decay.add("pos_emb")

        # validate that we considered every parameter
        param_dict = {pn: p for pn, p in self.named_parameters()}
        inter_params = decay & no_decay
        union_params = decay | no_decay
        assert (
            len(inter_params) == 0
        ), "parameters %s made it into both decay/no_decay sets!" % (str(inter_params),)
        assert (
            len(param_dict.keys() - union_params) == 0
        ), "parameters %s were not separated into either decay/no_decay set!" % (
            str(param_dict.keys() - union_params),
        )

        # create the pytorch optimizer object
        optim_groups = [
            {
                "params": [param_dict[pn] for pn in sorted(list(decay))],
                "weight_decay": train_config.weight_decay,
            },
            {
                "params": [param_dict[pn] for pn in sorted(list(no_decay))],
                "weight_decay": 0.0,
            },
        ]
        optimizer = torch.optim.AdamW(
            optim_groups, lr=train_config.learning_rate, betas=train_config.betas
        )
        return optimizer


    def forward(
        self,
        idx: torch.Tensor,
        targets: Optional[torch.Tensor] = None,
        mask: Optional[torch.Tensor] = None
    ) -> Union[Tuple[torch.Tensor, torch.Tensor, torch.Tensor], torch.Tensor]:
        """
        Forward pass through model.
        
        Args:
            idx: Input sequence [batch_size, seq_len, input_dim]
            targets: Target sequence [batch_size, seq_len, input_dim] (optional)
            mask: Mask for valid tokens [batch_size, seq_len] (optional) 

        Returns:
            If targets provided: (ce_loss, diffusion_loss, length_loss)
            If no targets: processed_logits [batch_size, seq_len, 5]
        """

        batch_size, seq_len, input_dim = idx.size()  # input_dim = 12 for {x, y, w, h, onehot(8)}
        assert seq_len <= self.block_size, "Cannot forward, model block size is exhausted."
        device = idx.device
        
        # Forward pass through the transformer
        token_embeddings = self.tok_emb(idx)  # [batch_size, seq_len, n_embd]
        position_embeddings = self.pos_emb[:, :seq_len, :]  # [1, seq_len, n_embd]
        x = self.drop(token_embeddings + position_embeddings)
        x = self.blocks(x)      # [batch_size, seq_len, n_embd]
        x = self.ln_f(x)        # [batch_size, seq_len, n_embd]

        # If we are given some desired targets also calculate the loss
        ce_loss = None
        diffusion_loss = None
        length_loss = None

        cont_dim = 4        # x, y, w, h
        disc_dim = 6        # one-hot class labels

        # Training mode (with targets)
        if targets is not None:
            targets_disc = targets[:, :, cont_dim:]    # [batch_size, seq_len, disc_dim]
            targets_cont = targets[:, :, :cont_dim]    # [batch_size, seq_len, cont_dim]
            logits = self.head(x)  # [batch_size, seq_len, disc_dim]
            
            # EOS enhancement 
            eos_enhancement = self.eos_mlp(x)       # [batch_size, max_length, 1]
            eos_enhancement = eos_enhancement.squeeze(-1) * self.eos_alpha  # [batch_size, max_length]
            logits[:, :, self.eos_token_idx] += eos_enhancement
    
            # Process discrete part (class prediction)
            logits_flat = logits.reshape(-1, disc_dim)  # [batch_size * seq_len, disc_dim]
            targets_disc_flat = targets_disc.reshape(-1, disc_dim)  # [batch_size * seq_len, disc_dim]
            # Apply mask to focus loss on valid tokens
            mask_flat = mask.reshape(-1)  # [batch_size * seq_len]

            logits_masked = logits_flat[mask_flat]
            targets_disc_masked = targets_disc_flat[mask_flat]
            


            # Calculate cross-entropy loss for discrete part
            target_indices = torch.argmax(targets_disc_masked, dim=-1)
            ce_loss = F.cross_entropy(
                logits_masked,
                target_indices
            )
            
            # Length Loss Calculation
            if self.length_loss_weight > 0:
                # Calculate probability of EOS token at each step 
                probs = F.softmax(logits, dim=-1)
                eos_probs = probs[:, :, self.eos_token_idx]     # [batch_size, seq_len]
                
                # Target length = number of non-pad tokens in gt
                target_lengths = mask.sum(dim=1).float()
                
                # Survival probability: prob of NOT ending before this step
                survival_probs = torch.cumprod(1.0 - eos_probs + 1e-8, dim=1)
                
                # Probability of ending at step t is P(eos at t) * P(not eos before t)
                # Shift survival_probs and prepend 1s for first step
                survival_shifted = torch.cat(
                    [torch.ones(batch_size, 1, device=device), survival_probs[:, :-1]],
                    dim=1
                )
                ending_probs = eos_probs * survival_shifted
                
                # Expected length is sum of (position * probability of ending at that position)
                positions = torch.arange(1, seq_len + 1, device=device, dtype=torch.float32).unsqueeze(0)
                expected_lengths = torch.sum(positions * ending_probs, dim=1)
                
                length_loss = F.mse_loss(expected_lengths, target_lengths)
            else:
                length_loss = torch.tensor(0.0, device=device)

            # Process continuous part (bounding box prediction)
            x_flat = x.reshape(-1, x.size(-1))  # [batch_size * seq_len, n_embd]
            
            x_flat = x_flat.repeat(self.diffusion_batch_mul, 1)  # [batch_size * seq_len * diffusion_batch_mul, n_embd]
            targets_cont_flat = targets_cont.reshape(-1, targets_cont.size(-1))
            targets_cont_flat = targets_cont_flat.repeat(self.diffusion_batch_mul, 1) # [batch_size * seq_len * diffusion_batch_mul, cont_dim=4]

            diffusion_loss = self.diffloss(z=x_flat, target=targets_cont_flat)

            return ce_loss, diffusion_loss, length_loss
        
        # Inference mode (without targets)
        else:
            # Process discrete part
            logits = self.head(x)  # [batch_size, seq_len, disc_dim]
            
            # EOS enhancement
            eos_enhancement = self.eos_mlp(x)
            eos_enhancement = eos_enhancement.squeeze(-1) * self.eos_alpha
            logits[:, :, self.eos_token_idx] += eos_enhancement
            
            probs_disc = F.softmax(logits, dim=-1)  # [batch_size, seq_len, disc_dim]

            _, top_disc = torch.topk(probs_disc, k=1, dim=-1)  # [batch_size, seq_len, 1]
            top_disc = top_disc.float()

            # Process continuous part (sample from diffusion network)
            flat_x = x.reshape(-1, x.size(-1))  # [batch_size * seq_len, n_embd]

            cont_samples = self.diffloss.sample(
                z=flat_x,
                temperature=1.0,
                cfg=1.0,
            )   # [batch_size * seq_len, cont_dim]

            sampled_cont = cont_samples.reshape(batch_size, seq_len, cont_dim)

            processed_logits = torch.cat(
                [
                    top_disc,  # [batch_size, seq_len, 1]
                    sampled_cont,  # [batch_size, seq_len, 4]
                ],
                dim=-1,
            )       # [batch_size, seq_len, 5]

            return processed_logits 
