from pytorch_lightning import LightningModule
from omegaconf import DictConfig
from omegaconf.errors import MissingMandatoryValue
import torch
import math
from typing import Union
from torch import nn
import torch.nn.functional as F
from torch.utils.data import DataLoader, Dataset
import logging
import numpy as np
from copy import deepcopy
from pytorch_lightning import Trainer
import ray
from ray import tune
from ray import ray_constants

from src.data import RealDatasetCollection, SyntheticDatasetCollection
from src.models.time_varying_model import TimeVaryingCausalModel
from src.models.utils import grad_reverse, BRTreatmentOutcomeHead, AlphaRise, clip_normalize_stabilized_weights
from src.models.utils_lstm import VariationalLSTM


logger = logging.getLogger(__name__)


class RMSN(TimeVaryingCausalModel):
    """
    Pytorch-Lightning implementation of Recurrent Marginal Structural Networks (RMSNs)
    (https://papers.nips.cc/paper/2018/hash/56e6a93212e4482d99c84a639d254b67-Abstract.html)
    """

    model_type = None  # Will be defined in subclasses
    possible_model_types = {'encoder', 'decoder', 'propensity_treatment', 'propensity_history'}
    tuning_criterion = None

    def __init__(self, args: DictConfig,
                 dataset_collection: Union[RealDatasetCollection, SyntheticDatasetCollection] = None,
                 autoregressive: bool = None,
                 has_vitals: bool = None,
                 bce_weights: np.array = None,
                 **kwargs):
        """
        Args:
            args: DictConfig of model hyperparameters
            dataset_collection: Dataset collection
            autoregressive: Flag of including previous outcomes to modelling
            has_vitals: Flag of vitals in dataset
            bce_weights: Re-weight BCE if used
            **kwargs: Other arguments
        """
        super().__init__(args, dataset_collection, autoregressive, has_vitals, bce_weights)

    def _init_specific(self, sub_args: DictConfig, encoder_r_size: int = None):
        # Encoder/decoder-specific parameters
        try:
            self.seq_hidden_units = sub_args.seq_hidden_units
            self.dropout_rate = sub_args.dropout_rate
            self.num_layer = sub_args.num_layer

            # Pytorch model init
            if self.seq_hidden_units is None or self.dropout_rate is None:
                raise MissingMandatoryValue()

            if self.model_type == 'decoder':
                self.memory_adapter = nn.Linear(encoder_r_size, self.seq_hidden_units)

            self.lstm = VariationalLSTM(self.input_size, self.seq_hidden_units, self.num_layer, self.dropout_rate)
            self.output_layer = nn.Linear(self.seq_hidden_units, self.output_size)

        except MissingMandatoryValue:
            logger.warning(f"{self.model_type} not fully initialised - some mandatory args are missing! "
                           f"(It's ok, if one will perform hyperparameters search afterward).")

    @staticmethod
    def set_hparams(model_args: DictConfig, new_args: dict, input_size: int, model_type: str):
        """
        Used for hyperparameter tuning and model reinitialisation
        :param model_args: Sub DictConfig, with encoder/decoder parameters
        :param new_args: New hyperparameters
        :param input_size: Input size of the model
        :param model_type: Submodel specification
        """
        sub_args = model_args[model_type]
        sub_args.optimizer.learning_rate = new_args['learning_rate']
        sub_args.batch_size = new_args['batch_size']
        sub_args.seq_hidden_units = int(input_size * new_args['seq_hidden_units'])
        sub_args.dropout_rate = new_args['dropout_rate']
        sub_args.num_layer = new_args['num_layer']
        sub_args.max_grad_norm = new_args['max_grad_norm']

    def get_propensity_scores(self, dataset: Dataset) -> np.array:
        logger.info(f'Propensity scores for {dataset.subset_name}.')
        if self.model_type == 'propensity_treatment' or self.model_type == 'propensity_history':
            data_loader = DataLoader(dataset, batch_size=self.hparams.dataset.val_batch_size, shuffle=False)
            propensity_scores = torch.cat(self.trainer.predict(self, data_loader))
        else:
            raise NotImplementedError()
        return propensity_scores.numpy()


class RMSNPropensityNetworkTreatment(RMSN):

    model_type = 'propensity_treatment'
    tuning_criterion = 'bce'

    def __init__(self, args: DictConfig,
                 dataset_collection: Union[RealDatasetCollection, SyntheticDatasetCollection] = None,
                 autoregressive: bool = None,
                 has_vitals: bool = None,
                 bce_weights: np.array = None,
                 **kwargs):
        super().__init__(args, dataset_collection, autoregressive, has_vitals, bce_weights)
        self.input_size = self.dim_treatments
        logger.info(f'Input size of {self.model_type}: {self.input_size}')

        self.output_size = self.dim_treatments

        self._init_specific(args.model.propensity_treatment)
        self.save_hyperparameters(args)

    def prepare_data(self) -> None:
        # Datasets normalisation etc.
        if self.dataset_collection is not None and not self.dataset_collection.processed_data_encoder:
            assert self.hparams.dataset.treatment_mode == 'multilabel'  # Only binary multilabel regime possible
            self.dataset_collection.process_data_encoder()
        if self.bce_weights is None and self.hparams.exp.bce_weight:
            self._calculate_bce_weights()

    def forward(self, batch):
        prev_treatments = batch['prev_treatments']
        x = self.lstm(prev_treatments, init_states=None)
        treatment_pred = self.output_layer(x)
        return treatment_pred

    def training_step(self, batch, batch_ind):
        treatment_pred = self(batch)
        bce_loss = self.bce_loss(treatment_pred, batch['current_treatments'].double(), kind='predict')
        bce_loss = (batch['active_entries'].squeeze(-1) * bce_loss).sum() / batch['active_entries'].sum()
        self.log(f'{self.model_type}_bce_loss', bce_loss, on_epoch=True, on_step=False, sync_dist=True)
        return bce_loss
    
    def validation_step(self, batch, batch_ind):
        treatment_pred = self(batch)
        bce_loss = self.bce_loss(treatment_pred, batch['current_treatments'].double(), kind='predict')
        bce_loss = (batch['active_entries'].squeeze(-1) * bce_loss).sum() / batch['active_entries'].sum()
        self.log('val/loss', bce_loss, on_epoch=True, on_step=False, sync_dist=True)
        return bce_loss

    def predict_step(self, batch, batch_ind, dataset_idx=None):
        return F.sigmoid(self(batch)).cpu()


class RMSNPropensityNetworkHistory(RMSN):

    model_type = 'propensity_history'
    tuning_criterion = 'bce'

    def __init__(self, args: DictConfig,
                 dataset_collection: Union[RealDatasetCollection, SyntheticDatasetCollection] = None,
                 autoregressive: bool = None,
                 has_vitals: bool = None,
                 bce_weights: np.array = None,
                 **kwargs):
        super().__init__(args, dataset_collection, autoregressive, has_vitals, bce_weights)
        self.input_size = self.dim_treatments + self.dim_static_features
        self.input_size += self.dim_vitals if self.has_vitals else 0
        self.input_size += self.dim_outcome if self.autoregressive else 0
        self.test_robustness = args.exp.test_robustness
        logger.info(f'Input size of {self.model_type}: {self.input_size}')
        logger.info('%s', f'test_robustness: {self.test_robustness}')

        self.output_size = self.dim_treatments

        self._init_specific(args.model.propensity_history)
        self.save_hyperparameters(args)

    def prepare_data(self) -> None:
        # Datasets normalisation etc.
        if self.dataset_collection is not None and not self.dataset_collection.processed_data_encoder:
            assert self.hparams.dataset.treatment_mode == 'multilabel'  # Only binary multilabel regime possible
            self.dataset_collection.process_data_encoder()
        if self.bce_weights is None and self.hparams.exp.bce_weight:
            self._calculate_bce_weights()

    def forward(self, batch, detach_treatment=False):
        prev_treatments = batch['prev_treatments']
        vitals_or_prev_outputs = []
        #! vitals_or_prev_outputs.append(batch['vitals']) if self.has_vitals else None

        if self.has_vitals: 
            vitals = batch['vitals']
            if self.test_robustness : 
                remove_indices = [2, 5]
                # Create a mask for the indices to keep
                keep_indices = torch.tensor([i for i in range(vitals.shape[2]) if i not in remove_indices]).to(vitals.device)
                # Remove the specified dimensions
                vitals = vitals.index_select(2, keep_indices)
            
            vitals_or_prev_outputs.append(vitals)

        else: 
            vitals = None

        vitals_or_prev_outputs.append(batch['prev_outputs']) if self.autoregressive else None
        vitals_or_prev_outputs = torch.cat(vitals_or_prev_outputs, dim=-1)
        static_features = batch['static_features']

        x = torch.cat((prev_treatments, vitals_or_prev_outputs), dim=-1)
        x = torch.cat((x, static_features.unsqueeze(1).expand(-1, x.size(1), -1)), dim=-1)
        x = self.lstm(x, init_states=None)
        treatment_pred = self.output_layer(x)
        return treatment_pred

    def training_step(self, batch, batch_ind):
        treatment_pred = self(batch)
        bce_loss = self.bce_loss(treatment_pred, batch['current_treatments'].double(), kind='predict')
        bce_loss = (batch['active_entries'].squeeze(-1) * bce_loss).sum() / batch['active_entries'].sum()
        self.log(f'{self.model_type}_bce_loss', bce_loss, on_epoch=True, on_step=False, sync_dist=True)
        return bce_loss
    
    def validation_step(self, batch, batch_ind):
        treatment_pred = self(batch)
        bce_loss = self.bce_loss(treatment_pred, batch['current_treatments'].double(), kind='predict')
        bce_loss = (batch['active_entries'].squeeze(-1) * bce_loss).sum() / batch['active_entries'].sum()
        self.log('val/loss', bce_loss, on_epoch=True, on_step=False, sync_dist=True)
        return bce_loss

    def predict_step(self, batch, batch_ind, dataset_idx=None):
        return F.sigmoid(self(batch)).cpu()


class RMSNEncoder(RMSN):

    model_type = 'encoder'
    tuning_criterion = 'rmse'

    def __init__(self, args: DictConfig,
                 propensity_treatment: RMSNPropensityNetworkTreatment = None,
                 propensity_history: RMSNPropensityNetworkHistory = None,
                 dataset_collection: Union[RealDatasetCollection, SyntheticDatasetCollection] = None,
                 autoregressive: bool = None,
                 has_vitals: bool = None,
                 bce_weights: np.array = None,
                 **kwargs):
        super().__init__(args, dataset_collection, autoregressive, has_vitals, bce_weights)
        self.input_size = self.dim_treatments + self.dim_static_features
        self.input_size += self.dim_vitals if self.has_vitals else 0
        self.input_size += self.dim_outcome if self.autoregressive else 0
        self.test_robustness = args.exp.test_robustness
        logger.info(f'Input size of {self.model_type}: {self.input_size}')
        logger.info('%s', f'test_robustness: {self.test_robustness}')

        self.output_size = self.dim_outcome

        self.propensity_treatment = propensity_treatment
        self.propensity_history = propensity_history

        self._init_specific(args.model.encoder)
        self.save_hyperparameters(args)

    def prepare_data(self) -> None:
        # Datasets normalisation etc.
        if self.dataset_collection is not None and not self.dataset_collection.processed_data_encoder:
            self.dataset_collection.process_data_encoder()

        if self.dataset_collection is not None and 'sw_tilde_enc' not in self.dataset_collection.train_f.data:
            self.dataset_collection.process_propensity_train_f(self.propensity_treatment, self.propensity_history)
            self.dataset_collection.process_propensity_val_f(self.propensity_treatment, self.propensity_history)

            self.dataset_collection.train_f.data['sw_tilde_enc'] = clip_normalize_stabilized_weights(
                self.dataset_collection.train_f.data['stabilized_weights'],
                self.dataset_collection.train_f.data['active_entries'],
                multiple_horizons=False
            )
            self.dataset_collection.val_f.data['sw_tilde_enc'] = clip_normalize_stabilized_weights(
                self.dataset_collection.val_f.data['stabilized_weights'],
                self.dataset_collection.val_f.data['active_entries'],
                multiple_horizons=False
            )
            
        if self.bce_weights is None and self.hparams.exp.bce_weight:
            self._calculate_bce_weights()

    def forward(self, batch, detach_treatment=False):
        vitals_or_prev_outputs = []
        #! vitals_or_prev_outputs.append(batch['vitals']) if self.has_vitals else None
        
        if self.has_vitals: 
            vitals = batch['vitals']
            if self.test_robustness : 
                remove_indices = [2, 5]
                # Create a mask for the indices to keep
                keep_indices = torch.tensor([i for i in range(vitals.shape[2]) if i not in remove_indices]).to(vitals.device)
                # Remove the specified dimensions
                vitals = vitals.index_select(2, keep_indices)
            
            vitals_or_prev_outputs.append(vitals)

        else: 
            vitals = None

        vitals_or_prev_outputs.append(batch['prev_outputs']) if self.autoregressive else None
        vitals_or_prev_outputs = torch.cat(vitals_or_prev_outputs, dim=-1)
        static_features = batch['static_features']
        curr_treatments = batch['current_treatments']

        x = torch.cat((vitals_or_prev_outputs, curr_treatments), dim=-1)
        x = torch.cat((x, static_features.unsqueeze(1).expand(-1, x.size(1), -1)), dim=-1)
        r = self.lstm(x, init_states=None)
        outcome_pred = self.output_layer(r)
        return outcome_pred, r

    def training_step(self, batch, batch_ind):
        outcome_pred, _ = self(batch)
        mse_loss = F.mse_loss(outcome_pred, batch['outputs'], reduce=False)
        weighted_mse_loss = mse_loss * batch['sw_tilde_enc'].unsqueeze(-1)
        weighted_mse_loss = (batch['active_entries'] * weighted_mse_loss).sum() / batch['active_entries'].sum()
        self.log(f'{self.model_type}_mse_loss', weighted_mse_loss, on_epoch=True, on_step=False, sync_dist=True)
        return weighted_mse_loss
    
    def validation_step(self, batch, batch_ind):
        outcome_pred, _ = self(batch)
        mse_loss = F.mse_loss(outcome_pred, batch['outputs'], reduce=False)
        weighted_mse_loss = mse_loss * batch['sw_tilde_enc'].unsqueeze(-1)
        weighted_mse_loss = (batch['active_entries'] * weighted_mse_loss).sum() / batch['active_entries'].sum()
        self.log('val/loss', weighted_mse_loss, on_epoch=True, on_step=False, sync_dist=True)
        return weighted_mse_loss

    def predict_step(self, batch, batch_ind, dataset_idx=None):
        outcome_pred, r = self(batch)
        return outcome_pred.cpu(), r.cpu()

    def get_representations(self, dataset: Dataset) -> np.array:
        logger.info(f'Representations inference for {dataset.subset_name}.')
        # Creating Dataloader
        data_loader = DataLoader(dataset, batch_size=self.hparams.dataset.val_batch_size, shuffle=False)
        _, r = [torch.cat(arrs) for arrs in zip(*self.trainer.predict(self, data_loader))]
        return r.numpy()

    def get_predictions(self, dataset: Dataset) -> np.array:
        logger.info(f'Predictions for {dataset.subset_name}.')
        # Creating Dataloader
        data_loader = DataLoader(dataset, batch_size=self.hparams.dataset.val_batch_size, shuffle=False)
        outcome_pred, _ = [torch.cat(arrs) for arrs in zip(*self.trainer.predict(self, data_loader))]
        return outcome_pred.numpy()


class RMSNDecoder(RMSN):

    model_type = 'decoder'
    tuning_criterion = 'rmse'

    def __init__(self, args: DictConfig,
                 encoder: RMSNEncoder = None,
                 dataset_collection: Union[RealDatasetCollection, SyntheticDatasetCollection] = None,
                 encoder_r_size: int = None,
                 autoregressive: bool = None,
                 has_vitals: bool = None,
                 bce_weights: np.array = None,
                 **kwargs):
        super().__init__(args, dataset_collection, autoregressive, has_vitals, bce_weights)

        self.input_size = self.dim_treatments + self.dim_static_features + self.dim_outcome
        logger.info(f'Input size of {self.model_type}: {self.input_size}')

        self.output_size = self.dim_outcome

        self.encoder = encoder
        encoder_r_size = self.encoder.seq_hidden_units if encoder is not None else encoder_r_size

        self._init_specific(args.model.decoder, encoder_r_size=encoder_r_size)
        self.save_hyperparameters(args)

    def prepare_data(self) -> None:
        # Datasets normalisation etc.
        if self.dataset_collection is not None and not self.dataset_collection.processed_data_decoder:
            self.dataset_collection.process_data_decoder(self.encoder)
        if self.dataset_collection is not None and 'sw_tilde_dec' not in self.dataset_collection.train_f.data:
            self.dataset_collection.train_f.data['stabilized_weights'] = \
                np.cumprod(self.dataset_collection.train_f.data['stabilized_weights'], axis=-1)[:, 1:]
            self.dataset_collection.val_f.data['stabilized_weights'] = \
                np.cumprod(self.dataset_collection.val_f.data['stabilized_weights'], axis=-1)[:, 1:]
            self.dataset_collection.train_f.data['sw_tilde_dec'] = clip_normalize_stabilized_weights(
                self.dataset_collection.train_f.data['stabilized_weights'],
                self.dataset_collection.train_f.data['active_entries'],
                multiple_horizons=True
            )
            self.dataset_collection.val_f.data['sw_tilde_dec'] = clip_normalize_stabilized_weights(
                self.dataset_collection.val_f.data['stabilized_weights'],
                self.dataset_collection.val_f.data['active_entries'],
                multiple_horizons=True
            )
        if self.bce_weights is None and self.hparams.exp.bce_weight:
            self._calculate_bce_weights()

    def forward(self, batch, detach_treatment=False):
        curr_treatments = batch['current_treatments']
        prev_outputs = batch['prev_outputs']
        static_features = batch['static_features']
        init_states = batch['init_state']

        x = torch.cat((curr_treatments, prev_outputs), dim=-1)
        x = torch.cat((x, static_features.unsqueeze(1).expand(-1, x.size(1), -1)), dim=-1)
        x = self.lstm(x, init_states=self.memory_adapter(init_states))
        outcome_pred = self.output_layer(x)
        return outcome_pred

    def training_step(self, batch, batch_ind):
        outcome_pred = self(batch)
        mse_loss = F.mse_loss(outcome_pred, batch['outputs'], reduce=False)
        weighted_mse_loss = mse_loss * batch['sw_tilde_dec'].unsqueeze(-1)
        weighted_mse_loss = (batch['active_entries'] * weighted_mse_loss).sum() / batch['active_entries'].sum()
        self.log(f'{self.model_type}_mse_loss', weighted_mse_loss, on_epoch=True, on_step=False, sync_dist=True)
        return weighted_mse_loss
    
    def validation_step(self, batch, batch_ind):
        outcome_pred = self(batch)
        mse_loss = F.mse_loss(outcome_pred, batch['outputs'], reduce=False)
        weighted_mse_loss = mse_loss * batch['sw_tilde_dec'].unsqueeze(-1)
        weighted_mse_loss = (batch['active_entries'] * weighted_mse_loss).sum() / batch['active_entries'].sum()
        self.log('val/loss', weighted_mse_loss, on_epoch=True, on_step=False, sync_dist=True)
        return weighted_mse_loss

    def predict_step(self, batch, batch_ind, dataset_idx=None):
        return self(batch).cpu()

    def get_predictions(self, dataset: Dataset) -> np.array:
        logger.info(f'Predictions for {dataset.subset_name}.')
        data_loader = DataLoader(dataset, batch_size=self.hparams.dataset.val_batch_size, shuffle=False)
        outcome_pred = torch.cat(self.trainer.predict(self, data_loader))
        return outcome_pred.numpy()
