"""
Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
SPDX-License-Identifier: Apache-2.0
"""
import torch
import torch.nn.functional as F
from torch import nn

from .common import ConvBlock, manual_pad


class ResNet1D(nn.Module):
    """ResNet feature extractor implementation for use in MILLET. Same as original architecture."""

    def __init__(self, n_in_channels: int, out_channels: int, padding_mode: str = "replicate"):
        super().__init__()
        self.n_in_channels = n_in_channels
        self.instance_encoder = nn.Sequential(
            ResNetBlock(n_in_channels, 64, padding_mode=padding_mode),
            ResNetBlock(64, 128, padding_mode=padding_mode),
            ResNetBlock(128, out_channels, padding_mode=padding_mode),
        )
        self.gap = nn.AdaptiveAvgPool1d(1)

    def forward(self, x: torch.Tensor) -> torch.Tensor:
        # PyTorch doesn't like replicate padding if the input tensor is too small, so pad manually to min length
        min_len = 5
        x = x.float()
        x = x.transpose(1,2)
        if x.shape[-1] >= min_len:
            x = self.instance_encoder(x)
            x = self.gap(x)             # (B, C, 1)
            x = x.squeeze(-1)
            return x
        else:
            padded_x = manual_pad(x, min_len)
            x = self.instance_encoder(padded_x)
            x = self.gap(x)             # (B, C, 1)
            x = x.squeeze(-1)
            return x

class ResNetBlock(nn.Module):
    """ResNet block of three convolutional blocks with different kernel sizes."""

    def __init__(self, in_channels: int, out_channels: int, padding_mode: str = "replicate", dropout_p: float = 0.3) -> None:
        super().__init__()

        # Create layers
        layers = []
        for block_idx, kernel_size in enumerate([8, 5, 3]):
            in_c = in_channels if block_idx == 0 else out_channels
            include_relu = block_idx == 2
            conv_block = ConvBlock(in_c, out_channels, kernel_size, padding_mode, include_relu)
            layers.append(conv_block)
            layers.append(nn.Dropout(p=dropout_p))
        self.layers = nn.Sequential(*layers)

        # Create residual
        self.residual: nn.Module
        if in_channels != out_channels:
            residual_layers = [
                nn.Conv1d(
                    in_channels=in_channels,
                    out_channels=out_channels,
                    kernel_size=1,
                    padding="same",
                    padding_mode=padding_mode,
                ),
                nn.BatchNorm1d(num_features=out_channels),
            ]
            self.residual = nn.Sequential(*residual_layers)
        else:
            self.residual = nn.BatchNorm1d(num_features=out_channels)

    def forward(self, x: torch.Tensor) -> torch.Tensor:
        return F.relu(self.layers(x) + self.residual(x))
