# Copyright (c) ByteDance, Inc. and its affiliates.
# All rights reserved.
#
# This source code is licensed under the license found in the
# LICENSE file in the root directory of this source tree.

"""
Mostly copy-paste from DINO and timm library:
https://github.com/facebookresearch/dino
https://github.com/rwightman/pytorch-image-models/blob/master/timm/models/vision_transformer.py
"""

import math
import torch
import torch.nn as nn

from functools import partial
from timm.models import register_model
import warnings
from torch import Tensor

import torch.nn.functional as F
import numpy as np


def _no_grad_trunc_normal_(tensor, mean, std, a, b):
    # Cut & paste from PyTorch official master until it's in a few official releases - RW
    # Method based on https://people.sc.fsu.edu/~jburkardt/presentations/truncated_normal.pdf
    def norm_cdf(x):
        # Computes standard normal cumulative distribution function
        return (1. + math.erf(x / math.sqrt(2.))) / 2.

    if (mean < a - 2 * std) or (mean > b + 2 * std):
        warnings.warn("mean is more than 2 std from [a, b] in nn.init.trunc_normal_. "
                      "The distribution of values may be incorrect.",
                      stacklevel=2)

    with torch.no_grad():
        # Values are generated by using a truncated uniform distribution and
        # then using the inverse CDF for the normal distribution.
        # Get upper and lower cdf values
        l = norm_cdf((a - mean) / std)
        u = norm_cdf((b - mean) / std)

        # Uniformly fill tensor with values from [l, u], then translate to
        # [2l-1, 2u-1].
        tensor.uniform_(2 * l - 1, 2 * u - 1)

        # Use inverse cdf transform for normal distribution to get truncated
        # standard normal
        tensor.erfinv_()

        # Transform to proper mean, std
        tensor.mul_(std * math.sqrt(2.))
        tensor.add_(mean)

        # Clamp to ensure it's in the proper range
        tensor.clamp_(min=a, max=b)
        return tensor


def trunc_normal_(tensor, mean=0., std=1., a=-2., b=2.):
    # type: (Tensor, float, float, float, float) -> Tensor
    return _no_grad_trunc_normal_(tensor, mean, std, a, b)



def drop_path(x, drop_prob: float = 0., training: bool = False):
    if drop_prob == 0. or not training:
        return x
    keep_prob = 1 - drop_prob
    shape = (x.shape[0],) + (1,) * (x.ndim - 1)  # work with diff dim tensors, not just 2D ConvNets
    random_tensor = keep_prob + torch.rand(shape, dtype=x.dtype, device=x.device)
    random_tensor.floor_()  # binarize
    output = x.div(keep_prob) * random_tensor
    return output


class DropPath(nn.Module):
    """Drop paths (Stochastic Depth) per sample  (when applied in main path of residual blocks).
    """
    def __init__(self, drop_prob=None):
        super(DropPath, self).__init__()
        self.drop_prob = drop_prob

    def forward(self, x):
        return drop_path(x, self.drop_prob, self.training)


class Mlp(nn.Module):
    def __init__(self, in_features, hidden_features=None, out_features=None, act_layer=nn.GELU, drop=0.):
        super().__init__()
        out_features = out_features or in_features
        hidden_features = hidden_features or in_features
        self.fc1 = nn.Linear(in_features, hidden_features)
        self.act = act_layer()
        self.fc2 = nn.Linear(hidden_features, out_features)
        self.drop = nn.Dropout(drop)

    def forward(self, x):
        x = self.fc1(x)
        x = self.act(x)
        x = self.drop(x)
        x = self.fc2(x)
        x = self.drop(x)
        return x


class Attention(nn.Module):
    def __init__(self, dim, num_heads=8, qkv_bias=False, qk_scale=None, attn_drop=0., proj_drop=0.):
        super().__init__()
        self.num_heads = num_heads
        head_dim = dim // num_heads
        self.scale = qk_scale or head_dim ** -0.5

        self.qkv = nn.Linear(dim, dim * 3, bias=qkv_bias)
        self.attn_drop = nn.Dropout(attn_drop)
        self.proj = nn.Linear(dim, dim)
        self.proj_drop = nn.Dropout(proj_drop)

    def forward(self, x):
        B, N, C = x.shape
        qkv = self.qkv(x).reshape(B, N, 3, self.num_heads, C // self.num_heads).permute(2, 0, 3, 1, 4)
        q, k, v = qkv[0], qkv[1], qkv[2]

        attn = (q @ k.transpose(-2, -1)) * self.scale
        attn = attn.softmax(dim=-1)
        attn = self.attn_drop(attn)

        x = (attn @ v).transpose(1, 2).reshape(B, N, C)
        x = self.proj(x)
        x = self.proj_drop(x)
        return x, attn

class Block(nn.Module):
    def __init__(self, dim, num_heads, mlp_ratio=4., qkv_bias=False, qk_scale=None, drop=0., 
                 attn_drop=0., drop_path=0., act_layer=nn.GELU, norm_layer=nn.LayerNorm, init_values=0):
        super().__init__()
        self.norm1 = norm_layer(dim)
        self.attn = Attention(
            dim, num_heads=num_heads, qkv_bias=qkv_bias, qk_scale=qk_scale, attn_drop=attn_drop, proj_drop=drop)
        self.drop_path = DropPath(drop_path) if drop_path > 0. else nn.Identity()
        self.norm2 = norm_layer(dim)
        mlp_hidden_dim = int(dim * mlp_ratio)
        self.mlp = Mlp(in_features=dim, hidden_features=mlp_hidden_dim, act_layer=act_layer, drop=drop)

        if init_values > 0:
            self.gamma_1 = nn.Parameter(init_values * torch.ones((dim)), requires_grad=True)
            self.gamma_2 = nn.Parameter(init_values * torch.ones((dim)), requires_grad=True)
        else:
            self.gamma_1, self.gamma_2 = None, None

    def forward(self, x, return_attention=False):
        y, attn = self.attn(self.norm1(x))
        if return_attention:
            return attn
        if self.gamma_1 is None:
            x = x + self.drop_path(y)
            x = x + self.drop_path(self.mlp(self.norm2(x)))
        else:
            x = x + self.drop_path(self.gamma_1 * y)
            x = x + self.drop_path(self.gamma_2 * self.mlp(self.norm2(x)))
        return x

"""
STEM NETWORK IMPLEMENTATIONS ===========================================================================================================
"""  
def pairwise_distance_v2(proxies, x, squared=False):
    if squared:
        return (torch.cdist(x, proxies, p=2)) ** 2
    else:
        return torch.cdist(x, proxies, p=2)

def proxy_loss(proxies, img_emb, gt_imgs, scale) -> Tensor:
    """
    proxies: shape of (num_classes, dim)
    img_emb: shape of (num_imgs, dim)
    gt_imgs: shape of (num_imgs)
    """
    proxies_emb = scale * F.normalize(proxies, p=2, dim=-1)
    img_emb = scale * F.normalize(img_emb, p=2, dim=-1)

    img_dist = pairwise_distance_v2(proxies=proxies_emb, x=img_emb, squared=True)
    img_dist = img_dist * -1.0

    cross_entropy = nn.CrossEntropyLoss(reduction="mean")
    img_loss = cross_entropy(img_dist, gt_imgs)
    return img_loss


def ortho_proj_loss_fn_v2(features, labels, gamma_s, gamma_d, reverse_pos_pairs: bool, use_square: bool):
    """
    features: shape (b, num_tokens, d)
    labels: shape (num_tokens)
    gamma_s, gamma_d: lambda_s and lambda_d in E.q (2) and (3) in the paper
    reverse_pos_pairs: If true, we want each token to be orthogonal to all other tokens, regarless of their channels.
    """
    device = features.device
    #  features are normalized
    features = F.normalize(features, p=2, dim=-1)

    labels = labels[None, :, None]  # extend dims

    mask = torch.eq(labels, labels.transpose(-2, -1)).bool().to(device)
    eye = torch.eye(mask.shape[-2], mask.shape[-1]).bool().to(device).unsqueeze(0)

    mask_pos = mask.masked_fill(eye, 0).float()
    mask_neg = (~mask).float()
    dot_prod = torch.matmul(features, features.transpose(-2, -1))

    mask_pos_sum = mask_pos.sum(dim=(-2, -1)) + 1e-6
    mask_neg_sum = mask_neg.sum(dim=(-2, -1)) + 1e-6

    pos_pairs_mean = (mask_pos * dot_prod).sum(dim=(-2, -1)) / mask_pos_sum
    neg_pairs_mean = (mask_neg * dot_prod).sum(dim=(-2, -1)) / mask_neg_sum

    if use_square:
        neg_pairs_mean = neg_pairs_mean**2

    if reverse_pos_pairs:
        if use_square:
            pos_pairs_mean = pos_pairs_mean**2
        loss = gamma_s * pos_pairs_mean + gamma_d * neg_pairs_mean
    else:
        loss = gamma_s * (1.0 - pos_pairs_mean) + gamma_d * neg_pairs_mean
    return loss.mean()


from torch import tensor
from einops import repeat, rearrange
import random

class DiChaViTEmbed(nn.Module):
    """ Image to Patch Embedding with Shared Projections
    """
    def __init__(self, img_size=224, patch_size=16, in_chans=3, embed_dim=768, channel_budget=2):
        super().__init__()

        num_patches = (img_size // patch_size) * (img_size // patch_size)
        self.img_size = img_size
        self.patch_size = patch_size
        self.num_patches = num_patches

        self.channel_budget = channel_budget

        self.proj = nn.Conv2d(1, embed_dim, kernel_size=patch_size, stride=patch_size)

        self.use_channelvit_channels=True
        self.run_ortho_loss=True
        self.run_proxy_loss=True

        self.channel_emb_proxies = torch.nn.Parameter((torch.randn(in_chans, embed_dim) / 8))
        # nn.init.orthogonal_(self.channel_emb_proxies)

        self.channel_embed = nn.Parameter(torch.randn(in_chans, embed_dim))
        # self.channel_embed = nn.Embedding(in_chans, embed_dim)

        # initialization of weights is done in the ViT CLASS

        self.channel_scale = np.sqrt(1.0 / 0.07)
        self.reverse_pos_pairs=True
        
        # # JumpCP configurations: the very high Temp forces a uniform distribution. So channels arent really diverse. 
        # self.proxy_loss_lambda = 0.001
        # self.ortho_loss_v1_lambda=0.001
        # self.sampling_temperature=1000
        # self.gamma_s=1
        # self.gamma_d=4

        # CHAMMI configurations
        self.proxy_loss_lambda = 0.1
        self.ortho_loss_v1_lambda=1
        self.sampling_temperature=0.1
        self.gamma_s=0.5
        self.gamma_d=2

    
    def forward(self, x, cur_channels=None, first_channel_idx=None):

        channel_embed = self.channel_embed

        b, Cin, h, w = x.shape
        Cin_original = Cin

        if self.training and cur_channels==None:

            cur_channels = [0,1,2,3]
            Cin_new = self.channel_budget # we dont have budget to run a random number of channels! 

            # channel_embed = self.channel_embed(tensor(cur_channels, device=x.device))

            with torch.no_grad():

                if first_channel_idx is None:
                    first_channel_idx = random.randint(0, Cin - 1)

                channel_emb_norm = F.normalize(channel_embed, p=2, dim=-1)
                channel_embed_cosine = torch.einsum(
                    "c d, e d -> c e", channel_emb_norm, channel_emb_norm
                )
                ## get the cosine similarity between the first channel and the rest
                cosine_scores = channel_embed_cosine[first_channel_idx]

                scores = (1 - cosine_scores) / self.sampling_temperature
                ## make the dist more peaky
                prob = F.softmax(scores, dim=-1)

                ## sample Cin_new channels without replacement
                indices = torch.multinomial(prob, Cin_new, replacement=False)
                indices = indices.cpu().numpy().tolist()
                if first_channel_idx not in indices:
                    indices[-1] = first_channel_idx
                cur_channels = [cur_channels[i] for i in indices]

            Cin = Cin_new
            channels_idx=cur_channels

            x = x[:, channels_idx, :, :]
            channel_embed = channel_embed[channels_idx]

        elif cur_channels==None:
            cur_channels=[0,1,2,3]
            channels_idx=cur_channels

        else:
            channels_idx=cur_channels
            x = x[:, channels_idx, :, :]
            channel_embed = channel_embed[channels_idx]

        x = rearrange(x, 'b (Cin S) ... -> (b Cin) S ...', S=1, Cin=Cin)
        x = self.proj(x) 
        x = rearrange(x, '(b Cin) Cout ... -> b Cout Cin ...', Cin=Cin) # B Cout Cin H W

        n_patches = x.shape[3] * x.shape[4]
        token_labels = torch.arange(x.shape[2]).repeat_interleave(n_patches).to(x.device)
        x_reshaped = rearrange(x, "B Cout Cin H W -> B (Cin H W) Cout").clone()

        if self.run_ortho_loss:
            orthoproj_loss = ortho_proj_loss_fn_v2(
                x_reshaped,
                labels=token_labels,
                gamma_s=self.gamma_s,
                gamma_d=self.gamma_d,
                reverse_pos_pairs=self.reverse_pos_pairs,
                use_square=False,
            )
        else:
            orthoproj_loss = 0

        if self.run_proxy_loss:
            channel_gt = torch.eye(Cin, device=x.device)
            channel_emb_proxies = self.channel_emb_proxies[cur_channels]
            proxyloss = proxy_loss(channel_emb_proxies, channel_embed, channel_gt, scale=self.channel_scale)
        else:
            proxyloss = 0

        ortho_proxy_loss = (
            orthoproj_loss * self.ortho_loss_v1_lambda + proxyloss * self.proxy_loss_lambda
        )
        channel_embed = repeat(channel_embed, "Cin Cout -> B Cout Cin", B=x.shape[0])
        x += channel_embed.unsqueeze(-1).unsqueeze(-1)

        x = rearrange(x, 'b Cout Cin ... -> b Cin Cout ...')

        return x, Cin, ortho_proxy_loss, first_channel_idx

    



class PatchEmbed(nn.Module):
    """ Image to Patch Embedding
    """
    def __init__(self, img_size=224, patch_size=16, in_chans=3, embed_dim=768, groups=1):
        super().__init__()
        num_patches = (img_size // patch_size) * (img_size // patch_size)
        self.img_size = img_size
        self.patch_size = patch_size
        self.num_patches = num_patches

        self.proj = nn.Conv2d(in_chans, embed_dim, kernel_size=patch_size, stride=patch_size, groups=groups)
            
    def forward(self, x):
        B, C, H, W = x.shape
        return self.proj(x)
    


class VisionTransformer(nn.Module):
    """ Vision Transformer """
    def __init__(self, img_size=[224], patch_size=16, in_chans=3, num_classes=0, embed_dim=768, depth=12,
                 num_heads=12, mlp_ratio=4., qkv_bias=False, qk_scale=None, drop_rate=0., attn_drop_rate=0.,
                 drop_path_rate=0., norm_layer=partial(nn.LayerNorm, eps=1e-6), return_all_tokens=False, 
                 init_values=0, use_mean_pooling=False, masked_im_modeling=False, num_prefix_tokens=1, 
                 
                 channel_budget=2
                 ):
        super().__init__()
        self.num_features = self.embed_dim = embed_dim
        self.return_all_tokens = return_all_tokens



        self.patch_embed = DiChaViTEmbed(
                    img_size=img_size[0], patch_size=patch_size, in_chans=in_chans, embed_dim=embed_dim, 
                    channel_budget=channel_budget)

        num_patches = self.patch_embed.num_patches

        self.cls_token = nn.Parameter(torch.zeros(1, num_prefix_tokens, embed_dim))
        self.pos_embed = nn.Parameter(torch.zeros(1, num_patches + num_prefix_tokens, embed_dim))
        self.pos_drop = nn.Dropout(p=drop_rate)
        self.num_prefix_tokens = num_prefix_tokens

        dpr = [x.item() for x in torch.linspace(0, drop_path_rate, depth)]  # stochastic depth decay rule
        self.blocks = nn.ModuleList([
            Block(
                dim=embed_dim, num_heads=num_heads, mlp_ratio=mlp_ratio, qkv_bias=qkv_bias, qk_scale=qk_scale,
                drop=drop_rate, attn_drop=attn_drop_rate, drop_path=dpr[i], norm_layer=norm_layer, 
                init_values=init_values)
            for i in range(depth)])

        self.norm = nn.Identity() if use_mean_pooling else norm_layer(embed_dim)
        self.fc_norm = norm_layer(embed_dim) if use_mean_pooling else None
        # Classifier head
        self.head = nn.Linear(embed_dim, num_classes) if num_classes > 0 else nn.Identity()

        trunc_normal_(self.pos_embed, std=.02)
        trunc_normal_(self.cls_token, std=.02)
        self.apply(self._init_weights)

        # nn.init.orthogonal_(self.patch_embed.channel_embed.weight)
        nn.init.orthogonal_(self.patch_embed.channel_embed)

        # masked image modeling
        self.masked_im_modeling = masked_im_modeling
        if masked_im_modeling:
            self.masked_embed = nn.Parameter(torch.zeros(1, embed_dim))
            self.keep_mask=True
        else:
            self.keep_mask=False

        return

    def _init_weights(self, m):
        if isinstance(m, nn.Linear):
            trunc_normal_(m.weight, std=.02)
            if isinstance(m, nn.Linear) and m.bias is not None:
                nn.init.constant_(m.bias, 0)
        elif isinstance(m, nn.LayerNorm):
            nn.init.constant_(m.bias, 0)
            nn.init.constant_(m.weight, 1.0)

    def reset_mask_tokens(self, use_mim=False):

        if use_mim:
            self.masked_embed = self.masked_embed

        else:
            self.masked_embed = self.masked_embed.detach()

        return

    
    def interpolate_pos_encoding(self, x, w, h, c):
        # number of auxilary dimensions before the patches
        if not hasattr(self, "num_extra_tokens"):
            # backward compatibility
            num_extra_tokens = 1
        else:
            num_extra_tokens = self.num_prefix_tokens

        npatch = x.shape[1] - num_extra_tokens
        N = self.pos_embed.shape[1] - num_extra_tokens

        if npatch == N and w == h:
            return self.pos_embed

        class_pos_embed = self.pos_embed[:, :num_extra_tokens]
        patch_pos_embed = self.pos_embed[:, num_extra_tokens:]

        dim = x.shape[-1]
        w0 = w // self.patch_embed.patch_size
        h0 = h // self.patch_embed.patch_size
        # we add a small number to avoid floating point error in the interpolation
        # see discussion at https://github.com/facebookresearch/dino/issues/8
        w0, h0 = w0 + 0.1, h0 + 0.1
        patch_pos_embed = nn.functional.interpolate(
            patch_pos_embed.reshape(
                1, int(math.sqrt(N)), int(math.sqrt(N)), dim
            ).permute(0, 3, 1, 2),
            scale_factor=(w0 / math.sqrt(N), h0 / math.sqrt(N)),
            mode="bicubic",
        )
        assert (
            int(w0) == patch_pos_embed.shape[-2]
            and int(h0) == patch_pos_embed.shape[-1]
        )
        patch_pos_embed = patch_pos_embed.permute(0, 2, 3, 1).view(1, 1, -1, dim)

        # create copies of the positional embeddings for each channel
        patch_pos_embed = patch_pos_embed.expand(1, c, -1, dim).reshape(1, -1, dim)

        return torch.cat((class_pos_embed, patch_pos_embed), dim=1)

    def prepare_tokens(self, x, mask=None, first_channel_idx=None):
        B, nc, w, h = x.shape
        # patch linear embedding
        x, Cin, ortho_proxy_loss, first_channel_idx = self.patch_embed(x,  first_channel_idx=first_channel_idx)

        nc = Cin

        C_NEW = x.shape[1]

        x = rearrange(x, 'B Cin ... -> (B Cin) ...')

        # mask image modeling
        if mask is not None:
            mask = mask.unsqueeze(1).expand(-1, C_NEW, -1, -1)
            mask = rearrange(mask, 'B Cin ... -> (B Cin) ...')

            x = self.mask_model(x, mask)

        x = rearrange(x, '(B Cin) ... -> B Cin ...', Cin=C_NEW)
        x = rearrange(x, 'B Cin Cout ... -> B Cout Cin ...', Cin=C_NEW)

        x = x.flatten(2).transpose(1, 2)

        # add the [CLS] token to the embed patch tokens
        cls_tokens = self.cls_token.expand(B, -1, -1)
        x = torch.cat((cls_tokens, x), dim=1)

        # add positional encoding to each token
        x = x + self.interpolate_pos_encoding(x, w, h, nc)

        return self.pos_drop(x), ortho_proxy_loss, first_channel_idx

    def forward(self, x, return_all_tokens=None, mask=None, first_channel_idx=None):
        # mim
        if self.masked_im_modeling:
            assert mask is not None
            x, ortho_proxy_loss, first_channel_idx = self.prepare_tokens(x, mask=mask, first_channel_idx=first_channel_idx)
        else:
            x, ortho_proxy_loss, first_channel_idx = self.prepare_tokens(x, first_channel_idx=first_channel_idx)

        for blk in self.blocks:
            x = blk(x)

        # print(x.mean(), x.var())

        x = self.norm(x)
        if self.fc_norm is not None:
            x[:, 0] = self.fc_norm(x[:, 1:, :].mean(1))
        
        return_all_tokens = self.return_all_tokens if \
            return_all_tokens is None else return_all_tokens
        if return_all_tokens:
            return x, ortho_proxy_loss, first_channel_idx
        
        return x[:, 0]


    def get_last_selfattention(self, x):
        x = self.prepare_tokens(x)
        for i, blk in enumerate(self.blocks):
            if i < len(self.blocks) - 1:
                x = blk(x)
            else:
                # return attention of the last block
                return blk(x, return_attention=True)

    def get_intermediate_layers(self, x, n=1):
        x = self.prepare_tokens(x)
        # we return the output tokens from the `n` last blocks
        output = []
        for i, blk in enumerate(self.blocks):
            x = blk(x)
            if len(self.blocks) - i <= n:
                output.append(self.norm(x))
        return output
        
    def get_num_layers(self):
        return len(self.blocks)

    def mask_model(self, x, mask):
        if self.keep_mask:
            x.permute(0, 2, 3, 1)[mask, :] = self.masked_embed.to(x.dtype)
        else:
            x.permute(0, 2, 3, 1)[mask, :] += 0.1*self.masked_embed.to(x.dtype)
            
        return x

def vit_tiny(patch_size=16, **kwargs):
    model = VisionTransformer(
        patch_size=patch_size, embed_dim=192, depth=12, num_heads=3, mlp_ratio=4,
        qkv_bias=True, **kwargs)
    return model

def vit_small(patch_size=16, **kwargs):
    model = VisionTransformer(
        patch_size=patch_size, embed_dim=384, depth=12, num_heads=6, mlp_ratio=4,
        qkv_bias=True, **kwargs)
    return model

def vit_base(patch_size=16, **kwargs):
    model = VisionTransformer(
        patch_size=patch_size, embed_dim=768, depth=12, num_heads=12, mlp_ratio=4,
        qkv_bias=True, **kwargs)
    return model

def vit_large(patch_size=16, **kwargs):
    model = VisionTransformer(
        patch_size=patch_size, embed_dim=1024, depth=24, num_heads=16, mlp_ratio=4,
        qkv_bias=True, **kwargs)
    return model



if __name__ == '__main__':

    model = vit_base(masked_im_modeling=True, in_chans=4).cuda()

    total_params=0
    for p in model.parameters():
        total_params += p.numel()

    print(total_params)

    model.masked_im_modeling=True


    total_channels = 4
    channel_budget = 2

    #  we emulate a dataloader here. 
    img_list = []
    chn_list = []

    bs=8

    img = torch.randn(bs, 4, 224, 224).cuda()
    
    mask = torch.randint(0, 2, (bs, 14, 14), dtype=torch.bool).cuda()

    out = model(img, mask=mask)

