import math
import torch.nn as nn
import torch.nn.functional as F

from einops import rearrange
from timm.layers import trunc_normal_
from torch.nn.utils import spectral_norm


def Normalize(in_channels, num_groups=32):
    return nn.GroupNorm(num_groups=num_groups, num_channels=in_channels, eps=1e-6, affine=True)

class Upsample(nn.Module):
    def __init__(self, in_channels, with_conv):
        super().__init__()
        self.with_conv = with_conv
        if self.with_conv:
            self.conv = spectral_norm(nn.Conv2d(in_channels,
                                                in_channels,
                                                kernel_size=3,
                                                stride=1,
                                                padding=1))

    def forward(self, x):
        x = nn.functional.interpolate(x, scale_factor=2.0, mode="nearest")
        if self.with_conv:
            x = self.conv(x)
        return x


class Downsample(nn.Module):
    def __init__(self, in_channels, with_conv):
        super().__init__()
        self.with_conv = with_conv
        if self.with_conv:
            self.conv = spectral_norm(nn.Conv2d(in_channels,
                                                in_channels,
                                                kernel_size=3,
                                                stride=2,
                                                padding=0))

    def forward(self, x):
        if self.with_conv:
            pad = (0,1,0,1)
            x = nn.functional.pad(x, pad, mode="constant", value=0)
            x = self.conv(x)
        else:
            x = nn.functional.avg_pool2d(x, kernel_size=2, stride=2)
        return x


class ResnetBlock(nn.Module):
    def __init__(self, *, in_channels, out_channels=None, conv_shortcut=False,
                dropout):
        super().__init__()
        self.in_channels = in_channels
        out_channels = in_channels if out_channels is None else out_channels
        self.out_channels = out_channels
        self.use_conv_shortcut = conv_shortcut

        self.norm1 = Normalize(in_channels)
        self.conv1 = spectral_norm(nn.Conv2d(in_channels,
                                             out_channels,
                                             kernel_size=3,
                                             stride=1,
                                             padding=1))
        
        self.norm2 = Normalize(out_channels)
        self.dropout = nn.Dropout(dropout)
        self.conv2 = spectral_norm(nn.Conv2d(out_channels,
                                             out_channels,
                                             kernel_size=3,
                                             stride=1,
                                             padding=1))
        self.act = nn.SiLU()
        if self.in_channels != self.out_channels:
            if self.use_conv_shortcut:
                self.conv_shortcut = spectral_norm(nn.Conv2d(in_channels,
                                                             out_channels,
                                                             kernel_size=3,
                                                             stride=1,
                                                             padding=1))
                trunc_normal_(self.conv_shortcut.weight, std=0.02)
                nn.init.zeros_(self.conv_shortcut.bias)
            else:
                self.nin_shortcut = spectral_norm(nn.Conv2d(in_channels,
                                                            out_channels,
                                                            kernel_size=1,
                                                            stride=1,
                                                            padding=0))
                trunc_normal_(self.nin_shortcut.weight, std=0.02)
                nn.init.zeros_(self.nin_shortcut.bias)


    def forward(self, x):
        h = x
        h = self.norm1(h)
        h = self.act(h)
        h = self.conv1(h)

        h = self.norm2(h)
        h = self.act(h)
        h = self.dropout(h)
        h = self.conv2(h)

        if self.in_channels != self.out_channels:
            if self.use_conv_shortcut:
                x = self.conv_shortcut(x)
            else:
                x = self.nin_shortcut(x)

        return x+h


class MultiAttnBlock(nn.Module):
    def __init__(self, in_channels, num_heads, res):
        super().__init__()
        self.in_channels = in_channels

        self.norm = Normalize(in_channels)
        self.norm_out = Normalize(in_channels)
        self.qkv = spectral_norm(nn.Conv2d(in_channels,
                                           in_channels * 3,
                                           kernel_size=1, 
                                           stride=1, 
                                           padding=0))
        self.proj_out = spectral_norm(nn.Conv2d(in_channels, 
                                                in_channels, 
                                                kernel_size=1,
                                                stride=1, 
                                                padding=0))
        self.num_heads = num_heads
        self.head_dim = in_channels // num_heads
        self.res = res

    def forward(self, x):
        h = w = self.res
        nh = self.num_heads
        hd = self.head_dim
        
        h_ = self.norm(x)
        qkv = self.qkv(h_)
        qkv = rearrange(qkv, 'b (qkv heads c) h w -> b qkv heads (h w) c', heads=nh, qkv=3, c=hd)
        q, k, v = qkv.unbind(1)
        attn = F.scaled_dot_product_attention(q, k, v)
        attn = rearrange(attn, 'b heads (h w) c -> b (heads c) h w', heads=nh, h=h, w=w)
        h_ = self.proj_out(self.norm_out(attn))
        
        return x + h_


def make_attn(in_channels, num_heads, res, attn_type="vanilla"):
    assert attn_type in ["vanilla", "none", "multi_heads"], f'attn_type {attn_type} unknown'
    if attn_type == "multi_heads":
        return MultiAttnBlock(in_channels, num_heads, res)
    elif attn_type == "none":
        return nn.Identity()
    elif attn_type == "vanilla":
        return MultiAttnBlock(in_channels, 1, res)


class Encoder(nn.Module):
    def __init__(self, *, ch=128, out_ch=3, ch_mult=(1,2,4,4), num_res_blocks=2,
                 attn_resolutions=[], attn_heads=[4,], dropout=0.0, resamp_with_conv=True, in_channels=3,
                 resolution=32, z_channels=512, attn_type="multi_heads", double_out=True, 
                 **ignore_kwargs):
        super().__init__()
        self.ch = ch
        self.num_resolutions = len(ch_mult)
        self.num_res_blocks = num_res_blocks
        self.resolution = resolution
        self.in_channels = in_channels
        num_heads = attn_heads.copy()
        num_heads = list(reversed(num_heads))

        self.conv_in = spectral_norm(nn.Conv2d(in_channels,
                                               self.ch,
                                               kernel_size=3,
                                               stride=1,
                                               padding=1))

        curr_res = resolution
        in_ch_mult = (1,)+tuple(ch_mult)
        self.in_ch_mult = in_ch_mult
        self.down = nn.ModuleList()
        for i_level in range(self.num_resolutions):
            block = nn.ModuleList()
            attn = nn.ModuleList()
            block_in = ch*in_ch_mult[i_level]
            block_out = ch*ch_mult[i_level]
            if curr_res in attn_resolutions:
                nh = num_heads.pop()
            for i_block in range(self.num_res_blocks):
                block.append(ResnetBlock(in_channels=block_in,
                                         out_channels=block_out,
                                         dropout=dropout))
                block_in = block_out
                if curr_res in attn_resolutions:
                    attn.append(make_attn(block_in, nh, curr_res, attn_type=attn_type))
            down = nn.Module()
            down.block = block
            down.attn = attn
            if i_level != self.num_resolutions-1:
                down.downsample = Downsample(block_in, resamp_with_conv)
                curr_res = curr_res // 2
            self.down.append(down)

        self.mid = nn.Module()
        self.mid.block_1 = ResnetBlock(in_channels=block_in,
                                       out_channels=block_in,
                                       dropout=dropout)
        self.mid.attn_1 = make_attn(block_in, num_heads.pop(), curr_res, attn_type=attn_type)
        self.mid.block_2 = ResnetBlock(in_channels=block_in,
                                       out_channels=block_in,
                                       dropout=dropout)
        self.norm_out = Normalize(block_in)
        self.conv_out = spectral_norm(nn.Conv2d(block_in,
                                                z_channels * 2 if double_out else z_channels,
                                                kernel_size=1,
                                                stride=1,
                                                padding=0))
        
        trunc_normal_(self.conv_out.weight, std=0.02)
        self.conv_out.bias.data[z_channels:].fill_(10.0)
        self.conv_out.bias.data[:z_channels].normal_(0, 0.01)

    def forward(self, x):
        h = self.conv_in(x)
        for i_level in range(self.num_resolutions):
            for i_block in range(self.num_res_blocks):
                h = self.down[i_level].block[i_block](h)
                if len(self.down[i_level].attn) > 0:
                    h = self.down[i_level].attn[i_block](h)
            if i_level != self.num_resolutions-1:
                h = self.down[i_level].downsample(h)

        h = self.mid.block_1(h)
        h = self.mid.attn_1(h)
        h = self.mid.block_2(h)

        h = self.norm_out(h)
        h = self.conv_out(h)
        return h


class Decoder(nn.Module):
    def __init__(self, *, ch=128, out_ch=3, ch_mult=(1,2,4,4), num_res_blocks=2,
                 attn_resolutions=[], attn_heads=[4,], dropout=0.0, resamp_with_conv=True, in_channels=3,
                 resolution=32, z_channels=512, double_out=True,
                 attn_type="multi_heads", **ignorekwargs):
        super().__init__()
        self.ch = ch
        self.num_resolutions = len(ch_mult)
        self.num_res_blocks = num_res_blocks
        self.resolution = resolution
        self.in_channels = in_channels
        out_ch = out_ch * 2 if double_out else out_ch
        num_heads = attn_heads.copy()

        in_ch_mult = (1,)+tuple(ch_mult)
        block_in = ch*ch_mult[self.num_resolutions-1]
        curr_res = resolution // 2**(self.num_resolutions-1)
        self.res = curr_res
        self.scale = math.sqrt(curr_res * curr_res * z_channels)

        self.conv_in = spectral_norm(nn.Conv2d(z_channels,
                                               block_in,
                                               kernel_size=1,
                                               stride=1,
                                               padding=0))

        self.mid = nn.Module()
        self.mid.block_1 = ResnetBlock(in_channels=block_in,
                                       out_channels=block_in,
                                       dropout=dropout)
        self.mid.attn_1 = make_attn(block_in, num_heads.pop(), curr_res, attn_type=attn_type)
        self.mid.block_2 = ResnetBlock(in_channels=block_in,
                                       out_channels=block_in,
                                       dropout=dropout)

        self.up = nn.ModuleList()
        for i_level in reversed(range(self.num_resolutions)):
            block = nn.ModuleList()
            attn = nn.ModuleList()
            block_out = ch*ch_mult[i_level]
            if curr_res in attn_resolutions:
                nh = num_heads.pop()
            for i_block in range(self.num_res_blocks+1):
                block.append(ResnetBlock(in_channels=block_in,
                                         out_channels=block_out,
                                         dropout=dropout))
                block_in = block_out
                if curr_res in attn_resolutions:
                    attn.append(make_attn(block_in, nh, curr_res, attn_type=attn_type))
            up = nn.Module()
            up.block = block
            up.attn = attn
            if i_level != 0:
                up.upsample = Upsample(block_in, resamp_with_conv)
                curr_res = curr_res * 2
            self.up.insert(0, up)

        self.norm_out = Normalize(block_in)
        self.conv_out = nn.Conv2d(block_in,
                                  out_ch,
                                  kernel_size=3,
                                  stride=1,
                                  padding=1)
        nn.init.zeros_(self.conv_out.weight)
        nn.init.zeros_(self.conv_out.bias)

    def forward(self, z):
        z = z * self.scale
        h = self.conv_in(z)

        h = self.mid.block_1(h)
        h = self.mid.attn_1(h)
        h = self.mid.block_2(h)

        for i_level in reversed(range(self.num_resolutions)):
            for i_block in range(self.num_res_blocks+1):
                h = self.up[i_level].block[i_block](h)
                if len(self.up[i_level].attn) > 0:
                    h = self.up[i_level].attn[i_block](h)
            if i_level != 0:
                h = self.up[i_level].upsample(h)

        h = self.conv_out(h)
        return h
