# Copyright (c) 2021 - present / Neuralmagic, Inc. All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#    http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing,
# software distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

from typing import List, Tuple

import numpy as np
import tensorflow

from sparseml.keras.optim import PruningScheduler
from sparseml.keras.utils import keras


__all__ = [
    "DenseLayerCreator",
    "SequentialModelCreator",
    "MockPruningScheduler",
    "model_01",
    "mnist_model",
]


class MockPruningScheduler(PruningScheduler):
    def __init__(self, step_and_sparsity_pairs: List[Tuple]):
        self._org_pairs = step_and_sparsity_pairs
        self.step_and_sparsity_pairs = {
            step: sparsity for (step, sparsity) in step_and_sparsity_pairs
        }

    def should_prune(self, step: int):
        return step in self.step_and_sparsity_pairs

    def target_sparsity(self, step: int):
        update_ready = step in self.step_and_sparsity_pairs
        sparsity = self.step_and_sparsity_pairs[step] if update_ready else None
        return sparsity

    def get_config(self):
        return {
            "class_name": self.__class__.__name__,
            "step_and_sparsity_pairs": self._org_pairs,
        }


class DenseLayer(keras.layers.Dense):
    def __init__(self, weight: np.ndarray):
        super(DenseLayer, self).__init__(weight.shape[1], activation=None)

        self.kernel = self.add_weight(
            "kernel",
            shape=weight.shape,
            initializer=keras.initializers.Constant(weight),
            dtype=tensorflow.float32,
            trainable=False,
        )
        self.bias = self.add_weight(
            "bias",
            shape=(weight.shape[1],),
            initializer=keras.initializers.Constant(0.0),
            dtype=tensorflow.float32,
            trainable=False,
        )


class LayerCreator:
    def __init__(self, kernel, bias=None):
        self.kernel = kernel
        self.bias = bias


class DenseLayerCreator(LayerCreator):
    def __init__(self, name, kernel, bias=None):
        super(DenseLayerCreator, self).__init__(kernel, bias)
        self.name = name

    def __call__(self, delay_build=False):
        layer = keras.layers.Dense(
            self.kernel.shape[-1], activation=None, name=self.name
        )
        if not delay_build:
            layer.build((None, self.kernel.shape[0]))
            assert len(layer.get_weights()) == 2
            if self.bias is None:
                self.bias = np.zeros((self.kernel.shape[-1],))
            layer.set_weights([self.kernel, self.bias])
        return layer


class SequentialModelCreator:
    def __init__(self, layer_creators):
        self.layer_creators = layer_creators

    def __call__(self):
        model = keras.Sequential()
        for layer_creator in self.layer_creators:
            model.add(layer_creator(delay_build=True))
        input_shape = tensorflow.TensorShape(
            (None, self.layer_creators[0].kernel.shape[-2])
        )
        model.build(input_shape=input_shape)
        return model


def model_01():
    inputs = keras.Input(shape=(5))
    x = keras.layers.Dense(10, name="dense_01")(inputs)
    outputs = keras.layers.Dense(10, name="dense_02")(x)
    model = keras.Model(inputs=inputs, outputs=outputs)
    return model


def mnist_model():
    inputs = tensorflow.keras.Input(shape=(28, 28, 1), name="inputs")

    # Block 1
    x = tensorflow.keras.layers.Conv2D(16, 5, strides=1)(inputs)
    x = tensorflow.keras.layers.BatchNormalization()(x)
    x = tensorflow.keras.layers.ReLU()(x)

    # Block 2
    x = tensorflow.keras.layers.Conv2D(32, 5, strides=2)(x)
    x = tensorflow.keras.layers.BatchNormalization()(x)
    x = tensorflow.keras.layers.ReLU()(x)

    # Block 3
    x = tensorflow.keras.layers.Conv2D(64, 5, strides=1)(x)
    x = tensorflow.keras.layers.BatchNormalization()(x)
    x = tensorflow.keras.layers.ReLU()(x)

    # Block 4
    x = tensorflow.keras.layers.Conv2D(128, 5, strides=2)(x)
    x = tensorflow.keras.layers.BatchNormalization()(x)
    x = tensorflow.keras.layers.ReLU()(x)

    x = tensorflow.keras.layers.AveragePooling2D(pool_size=1)(x)
    x = tensorflow.keras.layers.Flatten()(x)
    outputs = tensorflow.keras.layers.Dense(10, activation="softmax", name="outputs")(x)

    return tensorflow.keras.Model(inputs=inputs, outputs=outputs)
