import math
import torch
import torch.nn as nn
import numpy as np
from torch.nn.modules.utils import _pair
from scipy import ndimage
import torch.nn.functional as F
from models import configs
from tool.create_orthgonal import create_precise_random_standard_orthogonal_matrix

ATTENTION_Q = "MultiHeadDotProductAttention_1/query"
ATTENTION_K = "MultiHeadDotProductAttention_1/key"
ATTENTION_V = "MultiHeadDotProductAttention_1/value"
ATTENTION_OUT = "MultiHeadDotProductAttention_1/out"
FC_0 = "MlpBlock_3/Dense_0"
FC_1 = "MlpBlock_3/Dense_1"
ATTENTION_NORM = "LayerNorm_0"
MLP_NORM = "LayerNorm_2"

CONFIGS = {
    'ViT-B_16': configs.get_b16_config(),
    'ViT-B_32': configs.get_b32_config(),
    'ViT-L_16': configs.get_l16_config(),
    'ViT-L_32': configs.get_l32_config(),
    'ViT-H_14': configs.get_h14_config(),
    'R50-ViT-B_16': configs.get_r50_b16_config(),
    'testing': configs.get_testing(),
}


def np2th(weights, conv=False):
    """Possibly convert HWIO to OIHW."""
    if conv:
        weights = weights.transpose([3, 2, 0, 1])
    return torch.from_numpy(weights)


class Lora_Linear(nn.Module):
    def __init__(self, size_in, size_out, bias=True, enable_lora=False, FFN=False, n=6):
        super(Lora_Linear, self).__init__()
        self.enable_lora = enable_lora
        self.FFN = FFN
        self.size_in = size_in
        self.size_out = size_out
        self.has_bias = bias
        self.mlp = nn.Linear(size_in, size_out, bias=bias)
        # self.rank = int(self.size_in / n)
        self.num = min(size_in, size_out)

        if self.enable_lora:
            self.scaling = 1.0

            # Calculate number of basis matrices
            n_basis = n
            self.rank = self.size_out // n_basis
            # Create frozen random basis matrices
            self.basis_a = nn.Parameter(
                torch.zeros(n_basis, self.size_in, self.rank),
                requires_grad=False
            )
            self.basis_b = nn.Parameter(
                torch.zeros(1, self.rank, self.size_out),
                requires_grad=False
            )

            # Initialize basis matrices
            torch.nn.init.kaiming_uniform_(self.basis_a, a=math.sqrt(5))
            torch.nn.init.kaiming_uniform_(self.basis_b, a=math.sqrt(5))

            # Normalize basis matrices
            self.basis_a.data = self.basis_a.data / self.basis_a.data.std()
            self.basis_b.data = self.basis_b.data / self.basis_b.data.std()

            # Learnable combination coefficients
            self.A = nn.Parameter(torch.ones(n_basis, self.rank) / n_basis)
            self.B = nn.Parameter(torch.zeros(n_basis, self.size_out))

        # def get_update(self, w: Optional[torch.Tensor] = None) -> torch.Tensor:
        #     """Get the Random LoRA update"""
        #     # Combine basis matrices with learned coefficients
        #     basis_combined = self.basis_a.permute(1, 0, 2).flatten(start_dim=1)
        #     coeff_combined = (self.A[:, :, None] * self.basis_b * self.B[:, None, :]).flatten(end_dim=-2)
        #
        #     update = basis_combined @ coeff_combined
        #     return update * self.scaling

        self._frozen_param()

    def _frozen_param(self):
        for param in self.mlp.parameters():
            param.requires_grad = False

    def forward(self, x, randA = None, randB=None):
        if self.enable_lora:
            result = self.mlp(x)
            basis_combined = self.basis_a.permute(1, 0, 2).flatten(start_dim=1)
            # print(self.A[:, :, None].size())
            # print(self.basis_b.size())
            # print(self.B[:, None, :].size())
            coeff_combined = (self.A[:, :, None] * self.basis_b * self.B[:, None, :]).flatten(end_dim=-2)
            update = basis_combined @ coeff_combined
            result += (x @ update) * self.scaling
            return result
        else:
            return self.mlp(x)


class LoraAttention(nn.Module):
    def __init__(self, config, vis, enable_lora=False):
        super(LoraAttention, self).__init__()
        self.vis = vis
        self.num_attention_heads = config.transformer["num_heads"]
        self.attention_head_size = int(config.hidden_size / self.num_attention_heads)
        self.all_head_size = self.num_attention_heads * self.attention_head_size

        self.query = Lora_Linear(config.hidden_size, self.all_head_size, bias=True, enable_lora=enable_lora)
        self.key = Lora_Linear(config.hidden_size, self.all_head_size, bias=True, enable_lora=enable_lora)
        self.value = Lora_Linear(config.hidden_size, self.all_head_size, bias=True, enable_lora=enable_lora)
        self.out = Lora_Linear(config.hidden_size, config.hidden_size, bias=True, enable_lora=enable_lora)
        self.attn_dropout = nn.Dropout(config.transformer["attention_dropout_rate"])
        self.proj_dropout = nn.Dropout(config.transformer["attention_dropout_rate"])
        self.softmax = nn.Softmax(dim=-1)

    def transpose_for_scores(self, x):
        new_x_shape = x.size()[:-1] + (self.num_attention_heads, self.attention_head_size)
        x = x.view(*new_x_shape)
        return x.permute(0, 2, 1, 3)

    def forward(self, hidden_states, randA=None, randB=None):
        mixed_query_layer = self.query(hidden_states, randA, randB)
        mixed_key_layer = self.key(hidden_states, randA, randB)
        mixed_value_layer = self.value(hidden_states, randA, randB)

        query_layer = self.transpose_for_scores(mixed_query_layer)
        key_layer = self.transpose_for_scores(mixed_key_layer)
        value_layer = self.transpose_for_scores(mixed_value_layer)

        attention_scores = torch.matmul(query_layer, key_layer.transpose(-1, -2))
        attention_scores = attention_scores / math.sqrt(self.attention_head_size)
        attention_probs = self.softmax(attention_scores)
        # weights = attention_probs if self.vis else None
        attention_probs = self.attn_dropout(attention_probs)

        context_layer = torch.matmul(attention_probs, value_layer)
        context_layer = context_layer.permute(0, 2, 1, 3).contiguous()
        new_context_layer_shape = context_layer.size()[:-2] + (self.all_head_size,)
        context_layer = context_layer.view(*new_context_layer_shape)
        attention_output = self.out(context_layer, randA, randB)
        attention_output = self.proj_dropout(attention_output)
        return attention_output  # , weights

    def get_l1_loss(self):
        loss = self.query.get_l1_loss() + self.value.get_l1_loss() + self.key.get_l1_loss() + self.out.get_l1_loss()
        return loss


class LoraMLP(nn.Module):
    def __init__(self, config, enable_lora=False):
        super(LoraMLP, self).__init__()

        self.fc1 = Lora_Linear(config.hidden_size, config.transformer["mlp_dim"], bias=True, enable_lora=False)
        self.fc2 = Lora_Linear(config.transformer["mlp_dim"], config.hidden_size, bias=True, enable_lora=False)

        self.act = nn.GELU()
        self.dropout = nn.Dropout(config.transformer["dropout_rate"])

    def forward(self, x, orth=None):
        x = self.act(self.fc1(x))
        if self.training:
            x = self.dropout(x)
        x = self.fc2(x)
        if self.training:
            x = self.dropout(x)
        return x



class LoraBlock(nn.Module):
    def __init__(self, config, vis, drop_path=0.0, enable_lora=False):
        super(LoraBlock, self).__init__()
        self.hidden_size = config.hidden_size
        self.attention_norm = nn.LayerNorm(config.hidden_size, eps=1e-6)
        self.ffn_norm = nn.LayerNorm(config.hidden_size, eps=1e-6)
        self.ffn = LoraMLP(config, enable_lora=enable_lora)
        self.attn = LoraAttention(config, vis, enable_lora=enable_lora)
        self.enable_lora = enable_lora
        self.drop_path1 = nn.Dropout(p=drop_path)
        self.drop_path2 = nn.Dropout(p=drop_path)


    def forward(self, x, randA = None, randB=None):
        if randA is None:
            x = x + self.drop_path1(self.attn(self.attention_norm(x)))
            x = x + self.drop_path2(self.ffn(self.ffn_norm(x)))
        else:
            x = x + self.drop_path1(self.attn(self.attention_norm(x), randA, randB))
            x = x + self.drop_path2(self.ffn(self.ffn_norm(x)))
        return x

    def _fc_load_weight(self, ROOT, Key, Weights, unit):
        mat_weights = Weights[ROOT + '/' + Key + '/' + "kernel"]
        mat_bias = Weights[ROOT + '/' + Key + '/' + "bias"]
        if Key == ATTENTION_OUT:
            mat_weights = mat_weights.reshape(-1, mat_weights.shape[-1])
        else:
            mat_weights = mat_weights.reshape(mat_weights.shape[0], -1)

        unit.mlp.weight.copy_(np2th(mat_weights).t())
        unit.mlp.bias.copy_(np2th(mat_bias).view(-1))

    def load_from(self, weights, n_block):
        ROOT = f"Transformer/encoderblock_{n_block}"
        with torch.no_grad():
            self._fc_load_weight(ROOT, ATTENTION_Q, weights, self.attn.query)
            self._fc_load_weight(ROOT, ATTENTION_K, weights, self.attn.key)
            self._fc_load_weight(ROOT, ATTENTION_V, weights, self.attn.value)
            self._fc_load_weight(ROOT, ATTENTION_OUT, weights, self.attn.out)
            self._fc_load_weight(ROOT, FC_0, weights, self.ffn.fc1)
            self._fc_load_weight(ROOT, FC_1, weights, self.ffn.fc2)

            self.attention_norm.weight.copy_(np2th(weights[ROOT + '/' + ATTENTION_NORM + '/' + "scale"]))
            self.attention_norm.bias.copy_(np2th(weights[ROOT + '/' + ATTENTION_NORM + '/' + "bias"]))
            self.ffn_norm.weight.copy_(np2th(weights[ROOT + '/' + MLP_NORM + '/' + "scale"]))
            self.ffn_norm.bias.copy_(np2th(weights[ROOT + '/' + MLP_NORM + '/' + "bias"]))

    def get_l1_loss(self):
        return self.attn.get_l1_loss()

class LoraEncoder(nn.Module):
    def __init__(self, config, vis, drop_path=0.0,enable_lora=True, enable_orth = True, rank_n=6):
        super(LoraEncoder, self).__init__()
        self.vis = vis
        self.enable_lora = enable_lora
        self.layer = nn.ModuleList()
        # self.lora_layer = nn.ModuleList()
        for n in range(config.transformer["num_layers"]):
            setattr(self, f"lora_layer_{n}", nn.ModuleList())
        self.encoder_norm = nn.LayerNorm(config.hidden_size, eps=1e-6)
        self.num_blocks = config.transformer["num_layers"]
        self.depth = config.transformer["num_layers"]
        dpr = [x.item() for x in torch.linspace(0, drop_path, self.num_blocks)]
        # fellow SSF
        for i in range(config.transformer["num_layers"]):
            self.layer.append(LoraBlock(config, vis, drop_path=dpr[i], enable_lora=enable_lora))

    def forward(self, hidden_states):
        for layer_block in self.layer:
            hidden_states = layer_block(hidden_states)

        encoded = self.encoder_norm(hidden_states)
        return encoded

    def get_l1_loss(self):
        loss = 0
        for layer_i in self.layer:
            loss = loss + layer_i.get_l1_loss()
        return loss

class LoraEmbeddings(nn.Module):
    def __init__(self, config, img_size, in_channels=3):
        super(LoraEmbeddings, self).__init__()
        self.hybrid = None
        img_size = _pair(img_size)

        patch_size = _pair(config.patches["size"])
        n_patches = (img_size[0] // patch_size[0]) * (img_size[1] // patch_size[1])
        self.hybrid = False

        self.patch_embeddings = nn.Conv2d(in_channels=in_channels,
                                          out_channels=config.hidden_size,
                                          kernel_size=patch_size,
                                          stride=patch_size)
        self.position_embeddings = nn.Parameter(torch.zeros(1, n_patches + 1, config.hidden_size))
        self.cls_token = nn.Parameter(torch.zeros(1, 1, config.hidden_size))

        self.dropout = nn.Dropout(config.transformer["dropout_rate"])

    def forward(self, x):
        B = x.size(0)
        cls_tokens = self.cls_token.expand(B, -1, -1)

        x = self.patch_embeddings(x)
        x = x.flatten(2)
        x = x.transpose(-1, -2)
        x = torch.cat((cls_tokens, x), dim=1)
        embeddings = x + self.position_embeddings

        return embeddings


class LoraHouseTransformer(nn.Module):
    def __init__(self, config, img_size, vis, drop_path=0.0, enable_lora=True):
        super(LoraHouseTransformer, self).__init__()
        self.embeddings = LoraEmbeddings(config, img_size=img_size)
        self.encoder = LoraEncoder(config, vis, drop_path=drop_path,enable_lora=enable_lora)
        self._frozen_param()

    def _frozen_param(self):
        for param in self.embeddings.parameters():
            param.requires_grad = False

    def forward(self, input_ids):
        embedding_output = self.embeddings(input_ids)
        encoded = self.encoder(embedding_output)
        return encoded

    def get_l1_loss(self):
        return self.encoder.get_l1_loss()

class ranldoraVisionTransformer(nn.Module):
    def __init__(self, config, img_size=224, num_classes=21843, zero_head=False, vis=False,  enable_lora=True,
                 drop_path=0.0):
        super(ranldoraVisionTransformer, self).__init__()
        self.num_classes = num_classes
        self.zero_head = zero_head
        self.classifier = config.classifier
        self.transformer = LoraHouseTransformer(config, img_size, vis, drop_path=drop_path, enable_lora=enable_lora)
        self.head = nn.Linear(config.hidden_size, num_classes)
        self.loss_fct = nn.CrossEntropyLoss()

    def get_parameters(self, lr, weight_decay):
        wd_params = []
        no_wd_params = []
        for name, param in self.named_parameters():
            if 'bias' in name or 'norm' in name:
                no_wd_params.append(param)
            # elif "r_house" in name:
            #     no_wd_params.append(param)
            else:
                wd_params.append(param)

        params = [
            {"params": wd_params, "lr": lr, "weight_decay": weight_decay},
            {"params": no_wd_params, "lr": lr, "weight_decay": 0.}
        ]

        return params

    def forward(self, x, labels=None):
        x = self.transformer(x)

        logits = self.head(x[:, 0])
        if labels is not None:
            loss = self.loss_fct(logits.view(-1, self.num_classes), labels.view(-1))
            return loss
        else:
            return logits

    def load_from(self, weights):
        with torch.no_grad():
            if self.zero_head:
                nn.init.zeros_(self.head.weight)
                nn.init.zeros_(self.head.bias)
            else:
                self.head.weight.copy_(np2th(weights["head/kernel"]).t())
                self.head.bias.copy_(np2th(weights["head/bias"]).t())

            self.transformer.embeddings.patch_embeddings.weight.copy_(np2th(weights["embedding/kernel"], conv=True))
            self.transformer.embeddings.patch_embeddings.bias.copy_(np2th(weights["embedding/bias"]))
            self.transformer.embeddings.cls_token.copy_(np2th(weights["cls"]))
            self.transformer.encoder.encoder_norm.weight.copy_(np2th(weights["Transformer/encoder_norm/scale"]))
            self.transformer.encoder.encoder_norm.bias.copy_(np2th(weights["Transformer/encoder_norm/bias"]))

            posemb = np2th(weights["Transformer/posembed_input/pos_embedding"])
            posemb_new = self.transformer.embeddings.position_embeddings
            if posemb.size() == posemb_new.size():
                self.transformer.embeddings.position_embeddings.copy_(posemb)
            else:
                print("load_pretrained: resized variant: %s to %s" % (posemb.size(), posemb_new.size()))
                ntok_new = posemb_new.size(1)

                if self.classifier == "token":
                    posemb_tok, posemb_grid = posemb[:, :1], posemb[0, 1:]
                    ntok_new -= 1
                else:
                    posemb_tok, posemb_grid = posemb[:, :0], posemb[0]

                gs_old = int(np.sqrt(len(posemb_grid)))
                gs_new = int(np.sqrt(ntok_new))
                print('load_pretrained: grid-size from %s to %s' % (gs_old, gs_new))
                posemb_grid = posemb_grid.reshape(gs_old, gs_old, -1)

                zoom = (gs_new / gs_old, gs_new / gs_old, 1)
                posemb_grid = ndimage.zoom(posemb_grid, zoom, order=1)
                posemb_grid = posemb_grid.reshape(1, gs_new * gs_new, -1)
                posemb = np.concatenate([posemb_tok, posemb_grid], axis=1)
                self.transformer.embeddings.position_embeddings.copy_(np2th(posemb))

            for bname, block in self.transformer.encoder.named_children():
                for uname, unit in block.named_children():
                    unit.load_from(weights, n_block=uname)

    def get_l1_loss(self):
        return self.transformer.get_l1_loss()