from __future__ import annotations

from typing import TYPE_CHECKING

import inferno
import lightning as L
from inferno.bnn import params

from . import _nn_hyperparameters
from ._nn_model import _NNModel

if TYPE_CHECKING:
    from jaxtyping import Float
    from torch import Tensor


class ResNet(_NNModel):

    def __init__(
        self,
        resnet_type: type[inferno.models.ResNet],
        resnet_architecture: str,
        out_size: int,
        parametrization: params.Parametrization,
        lr: float,
        momentum: float,
        nesterov: bool,
        weight_decay: float,
        pretrained: bool,
        freeze_pretrained_weights: bool,
        max_epochs: int,
    ) -> None:
        super().__init__(
            num_classes=out_size,
            lr=lr,
            momentum=momentum,
            nesterov=nesterov,
            weight_decay=weight_decay,
            max_epochs=max_epochs,
        )

        # Model
        if pretrained:
            self.model = resnet_type.from_pretrained_weights(
                out_size=out_size,
                architecture=resnet_architecture,
                parametrization=parametrization,
                cov=None,
                freeze=freeze_pretrained_weights,
            )
        else:
            self.model = resnet_type(
                out_size=out_size,
                architecture=resnet_architecture,
                parametrization=parametrization,
                cov=None,
            )
        self.save_hyperparameters(
            _nn_hyperparameters(
                lightning_module=self,
                architecture=resnet_type.__name__,
                out_size=out_size,
                lr=lr,
                momentum=momentum,
                nesterov=nesterov,
                weight_decay=weight_decay,
                max_epochs=max_epochs,
            ),
            logger=True,
        )


class ResNet18(ResNet):

    @classmethod
    def from_dataset(
        cls,
        dataset: L.LightningDataModule,
        parametrization: params.Parametrization,
        lr: float,
        momentum: float,
        nesterov: bool,
        weight_decay: float,
        max_epochs: int,
        pretrained: bool,
        freeze_pretrained_weights: bool,
        seed: int,
        root_dir: str,
    ):
        if dataset.__class__.__name__ in [
            "CIFAR10",
            "CIFAR100",
            "TinyImageNet",
        ]:
            resnet_architecture = "cifar"
        elif dataset.__class__.__name__ in ["ImageNet"]:
            resnet_architecture = "imagenet"
        else:
            raise NotImplementedError()

        return cls(
            resnet_type=inferno.models.ResNet18,
            resnet_architecture=resnet_architecture,
            out_size=dataset.num_classes,
            parametrization=parametrization,
            lr=lr,
            momentum=momentum,
            nesterov=nesterov,
            weight_decay=weight_decay,
            max_epochs=max_epochs,
            pretrained=pretrained,
            freeze_pretrained_weights=freeze_pretrained_weights,
        )


class ResNet34(ResNet):

    @classmethod
    def from_dataset(
        cls,
        dataset: L.LightningDataModule,
        parametrization: params.Parametrization,
        lr: float,
        momentum: float,
        nesterov: bool,
        weight_decay: float,
        max_epochs: int,
        pretrained: bool,
        freeze_pretrained_weights: bool,
        seed: int,
        root_dir: str,
    ):
        if dataset.__class__.__name__ in [
            "CIFAR10",
            "CIFAR100",
            "TinyImageNet",
        ]:
            resnet_architecture = "cifar"
        elif dataset.__class__.__name__ in ["ImageNet"]:
            resnet_architecture = "imagenet"
        else:
            raise NotImplementedError()

        return cls(
            resnet_type=inferno.models.ResNet34,
            resnet_architecture=resnet_architecture,
            out_size=dataset.num_classes,
            parametrization=parametrization,
            lr=lr,
            momentum=momentum,
            nesterov=nesterov,
            weight_decay=weight_decay,
            max_epochs=max_epochs,
            pretrained=pretrained,
            freeze_pretrained_weights=freeze_pretrained_weights,
        )


class ResNet50(ResNet):

    @classmethod
    def from_dataset(
        cls,
        dataset: L.LightningDataModule,
        parametrization: params.Parametrization,
        lr: float,
        momentum: float,
        nesterov: bool,
        weight_decay: float,
        max_epochs: int,
        pretrained: bool,
        freeze_pretrained_weights: bool,
        seed: int,
        root_dir: str,
    ):
        if dataset.__class__.__name__ in [
            "CIFAR10",
            "CIFAR100",
            "TinyImageNet",
        ]:
            resnet_architecture = "cifar"
        elif dataset.__class__.__name__ in ["ImageNet"]:
            resnet_architecture = "imagenet"
        else:
            raise NotImplementedError()

        return cls(
            resnet_type=inferno.models.ResNet50,
            resnet_architecture=resnet_architecture,
            out_size=dataset.num_classes,
            parametrization=parametrization,
            lr=lr,
            momentum=momentum,
            nesterov=nesterov,
            weight_decay=weight_decay,
            max_epochs=max_epochs,
            pretrained=pretrained,
            freeze_pretrained_weights=freeze_pretrained_weights,
        )


class ResNet101(ResNet):

    @classmethod
    def from_dataset(
        cls,
        dataset: L.LightningDataModule,
        parametrization: params.Parametrization,
        lr: float,
        momentum: float,
        nesterov: bool,
        weight_decay: float,
        max_epochs: int,
        pretrained: bool,
        freeze_pretrained_weights: bool,
        seed: int,
        root_dir: str,
    ):
        if dataset.__class__.__name__ in [
            "CIFAR10",
            "CIFAR100",
            "TinyImageNet",
        ]:
            resnet_architecture = "cifar"
        elif dataset.__class__.__name__ in ["ImageNet"]:
            resnet_architecture = "imagenet"
        else:
            raise NotImplementedError()

        return cls(
            resnet_type=inferno.models.ResNet101,
            resnet_architecture=resnet_architecture,
            out_size=dataset.num_classes,
            parametrization=parametrization,
            lr=lr,
            momentum=momentum,
            nesterov=nesterov,
            weight_decay=weight_decay,
            max_epochs=max_epochs,
            pretrained=pretrained,
            freeze_pretrained_weights=freeze_pretrained_weights,
        )


class WideResNet50(ResNet):

    @classmethod
    def from_dataset(
        cls,
        dataset: L.LightningDataModule,
        parametrization: params.Parametrization,
        lr: float,
        momentum: float,
        nesterov: bool,
        weight_decay: float,
        max_epochs: int,
        pretrained: bool,
        freeze_pretrained_weights: bool,
        seed: int,
        root_dir: str,
    ):
        if dataset.__class__.__name__ in [
            "CIFAR10",
            "CIFAR100",
            "TinyImageNet",
        ]:
            resnet_architecture = "cifar"
        elif dataset.__class__.__name__ in ["ImageNet"]:
            resnet_architecture = "imagenet"
        else:
            raise NotImplementedError()

        return cls(
            resnet_type=inferno.models.WideResNet50,
            resnet_architecture=resnet_architecture,
            out_size=dataset.num_classes,
            parametrization=parametrization,
            lr=lr,
            momentum=momentum,
            nesterov=nesterov,
            weight_decay=weight_decay,
            max_epochs=max_epochs,
            pretrained=pretrained,
            freeze_pretrained_weights=freeze_pretrained_weights,
        )


class WideResNet101(ResNet):

    @classmethod
    def from_dataset(
        cls,
        dataset: L.LightningDataModule,
        parametrization: params.Parametrization,
        lr: float,
        momentum: float,
        nesterov: bool,
        weight_decay: float,
        max_epochs: int,
        pretrained: bool,
        freeze_pretrained_weights: bool,
        seed: int,
        root_dir: str,
    ):
        if dataset.__class__.__name__ in [
            "CIFAR10",
            "CIFAR100",
            "TinyImageNet",
        ]:
            resnet_architecture = "cifar"
        elif dataset.__class__.__name__ in ["ImageNet"]:
            resnet_architecture = "imagenet"
        else:
            raise NotImplementedError()

        return cls(
            resnet_type=inferno.models.WideResNet101,
            resnet_architecture=resnet_architecture,
            out_size=dataset.num_classes,
            parametrization=parametrization,
            lr=lr,
            momentum=momentum,
            nesterov=nesterov,
            weight_decay=weight_decay,
            max_epochs=max_epochs,
            pretrained=pretrained,
            freeze_pretrained_weights=freeze_pretrained_weights,
        )
