import torch
import torch.nn as nn
import torch.nn.functional as F
from geomloss import SamplesLoss
import torch.distributions as dist
from torch.distributions import Normal

def batch_mean_except_first(x: torch.Tensor) -> torch.Tensor:
    if x.dim() < 2:
        raise ValueError("Input tensor requires at least 2 dimensions")
    #Flatten all dimensions except batch to second dimension
    flat = x.reshape(x.size(0), -1)  #shape = (batch, number of elements for aggregation)
    #Averaging the second dimension after flattening and maintaining the dimension
    return flat.mean(dim=1, keepdim=True)

class GenerativeModel(nn.Module):
    def __init__(self, config):
        super(GenerativeModel, self).__init__()
        #Basic dimension parameters
        self.z_dim = config['model']['z_dim']
        self.hidden_dim = config['model']['generative']['hidden_dim']
        self.num_layers = config['model']['generative']['num_layers']
        self.treatment_size = config['dataset']['treatment_size']
        self.treatment_hidden_dim = config['model']['generative']['treatment_hidden_dim']
        self.output_dim = config['dataset']['output_size']
        self.input_size = config['dataset']['input_size']
        self.static_size = config['dataset']['static_size']
        self.autoregressive = config['dataset']['autoregressive']
        self.dropout = config['exp']['dropout']
        self.beta_bound = config['exp']['beta_bound']
        self.config = config
        self.predict_X = config['dataset']['predict_X']
        self.input_size = config['dataset']['input_size']

        self.reverse_action_encoder = nn.LSTM(
            input_size=self.treatment_size,
            hidden_size=self.treatment_hidden_dim,
            num_layers=1,
            batch_first=True,
            dropout=self.dropout
        )

        self.action_encoder = nn.LSTM(
            input_size=self.treatment_size,
            hidden_size=self.treatment_hidden_dim,
            num_layers=1,
            batch_first=True,
            dropout=self.dropout
        )

        self.action_decoder = nn.Sequential()
        self.hidden_action_decoder = config['model']['generative']['hidden_action_decoder']
        output_size = self.treatment_size 
        if -1 not in self.hidden_action_decoder:
            for i in range(len(self.hidden_action_decoder)):
                if i == 0:
                    self.action_decoder.add_module('fc{}'.format(i), nn.Linear(self.treatment_hidden_dim + self.z_dim, self.hidden_action_decoder[i]))
                else:
                    self.action_decoder.add_module('elu{}'.format(i), nn.ELU())
                    self.action_decoder.add_module('fc{}'.format(i), nn.Linear(self.hidden_action_decoder[i-1], self.hidden_action_decoder[i]))
            self.action_decoder.add_module('elu{}'.format(len(self.hidden_action_decoder)), nn.ELU())
            self.action_decoder.add_module('fc{}'.format(len(self.hidden_action_decoder)), nn.Linear(self.hidden_action_decoder[-1], output_size))
        else:
            self.action_decoder.add_module('fc{}'.format(1), nn.Linear(self.treatment_hidden_dim + self.z_dim, output_size))

        self.action_decoder_beta = nn.Sequential()
        self.hidden_action_decoder = config['model']['generative']['hidden_action_decoder']
        output_size = self.treatment_size * 2
        if -1 not in self.hidden_action_decoder:
            for i in range(len(self.hidden_action_decoder)):
                if i == 0:
                    self.action_decoder_beta.add_module('fc{}'.format(i), nn.Linear(self.treatment_hidden_dim + self.z_dim, self.hidden_action_decoder[i]))
                else:
                    self.action_decoder_beta.add_module('elu{}'.format(i), nn.ELU())
                    self.action_decoder_beta.add_module('fc{}'.format(i), nn.Linear(self.hidden_action_decoder[i-1], self.hidden_action_decoder[i]))
            self.action_decoder_beta.add_module('elu{}'.format(len(self.hidden_action_decoder)), nn.ELU())
            self.action_decoder_beta.add_module('fc{}'.format(len(self.hidden_action_decoder)), nn.Linear(self.hidden_action_decoder[-1], output_size))
        else:
            self.action_decoder_beta.add_module('fc{}'.format(1), nn.Linear(self.treatment_hidden_dim + self.z_dim, output_size))

        self.action_decoder_step = nn.Sequential()
        self.hidden_action_decoder_step = config['model']['generative']['hidden_action_decoder_step']
        output_size = self.treatment_size * 2
        if -1 not in self.hidden_action_decoder_step:
            for i in range(len(self.hidden_action_decoder_step)):
                if i == 0:
                    self.action_decoder_step.add_module('fc{}'.format(i), nn.Linear(self.z_dim, self.hidden_action_decoder_step[i]))
                else:
                    self.action_decoder_step.add_module('elu{}'.format(i), nn.ELU())
                    self.action_decoder_step.add_module('fc{}'.format(i), nn.Linear(self.hidden_action_decoder_step[i-1], self.hidden_action_decoder_step[i]))
            self.action_decoder_step.add_module('elu{}'.format(len(self.hidden_action_decoder_step)), nn.ELU())
            self.action_decoder_step.add_module('fc{}'.format(len(self.hidden_action_decoder_step)), nn.Linear(self.hidden_action_decoder_step[-1], output_size))
        else:
            self.action_decoder_step.add_module('fc{}'.format(1), nn.Linear(self.z_dim, output_size))

        
        if self.static_size > 0:
            input_size = self.input_size + self.static_size + self.treatment_size
        else:
            input_size = self.input_size + self.treatment_size
        if self.autoregressive:
            input_size += self.output_dim
        if self.predict_X:
            input_size += self.input_size

        self.lstm_history = nn.LSTM(
            input_size=input_size,
            hidden_size=self.z_dim,
            num_layers=self.num_layers,
            batch_first=True,
            dropout=self.dropout
        )
        
        #Store hidden state
        self.hidden_state = None
        self.cell_state = None

        self.lstm_input_dim = self.z_dim + self.treatment_hidden_dim
        # self.lstm_input_dim = self.z_dim + self.treatment_size
        self.lstm = nn.LSTM(
            input_size=self.lstm_input_dim,
            hidden_size=self.hidden_dim,
            num_layers=self.num_layers,
            batch_first=True,
            dropout=self.dropout
        )

        input_size = self.z_dim + self.treatment_size
        self.hidden_lstm = nn.LSTM(
            input_size=input_size,
            hidden_size=self.z_dim,
            num_layers=self.num_layers,
            batch_first=True,
            dropout=self.dropout
        )

        input_size = self.z_dim 
        self.hidden_lstm_reverse = nn.LSTM(
            input_size=input_size,
            hidden_size=self.z_dim,
            num_layers=self.num_layers,
            batch_first=True,
            dropout=self.dropout
        )

        #Mean Network
        self.hiddens_F_mu = config['model']['generative']['hiddens_F_mu']
        self.fc_mu = nn.Sequential()
        # input_size = self.hidden_dim + self.treatment_size
        input_size = self.hidden_dim + self.treatment_hidden_dim

        if -1 not in self.hiddens_F_mu:
            for i in range(len(self.hiddens_F_mu)):
                if i == 0:
                    self.fc_mu.add_module('fc{}'.format(i), nn.Linear(input_size, self.hiddens_F_mu[i]))
                else:
                    self.fc_mu.add_module('elu{}'.format(i), nn.ELU())
                    self.fc_mu.add_module('fc{}'.format(i), nn.Linear(self.hiddens_F_mu[i-1], self.hiddens_F_mu[i]))
            self.fc_mu.add_module('elu{}'.format(len(self.hiddens_F_mu)), nn.ELU())
            self.fc_mu.add_module('fc{}'.format(len(self.hiddens_F_mu)), nn.Linear(self.hiddens_F_mu[-1], self.z_dim))
        else:
            self.fc_mu.add_module('fc{}'.format(1), nn.Linear(input_size, self.z_dim))
        
        self.hiddens_F_logvar = config['model']['generative']['hiddens_F_logvar']
        self.fc_logvar = nn.Sequential()
        if -1 not in self.hiddens_F_logvar:
            for i in range(len(self.hiddens_F_logvar)):
                if i == 0:
                    self.fc_logvar.add_module('fc{}'.format(i), nn.Linear(input_size, self.hiddens_F_logvar[i]))
                else:
                    self.fc_logvar.add_module('elu{}'.format(i), nn.ELU())
                    self.fc_logvar.add_module('fc{}'.format(i), nn.Linear(self.hiddens_F_logvar[i-1], self.hiddens_F_logvar[i]))
            self.fc_logvar.add_module('elu{}'.format(len(self.hiddens_F_logvar)), nn.ELU())
            self.fc_logvar.add_module('fc{}'.format(len(self.hiddens_F_logvar)), nn.Linear(self.hiddens_F_logvar[-1], self.z_dim))
        else:
            self.fc_logvar.add_module('fc{}'.format(1), nn.Linear(input_size, self.z_dim))
        
        
        input_size = self.z_dim
        self.hiddens_decoder = config['model']['generative']['hiddens_decoder']
        self.decoder = nn.Sequential()
        if -1 not in self.hiddens_decoder:
            for i in range(len(self.hiddens_decoder)):
                if i == 0:
                    self.decoder.add_module('fc{}'.format(i), nn.Linear(input_size, self.hiddens_decoder[i]))
                else:
                    self.decoder.add_module('elu{}'.format(i), nn.ELU())
                    self.decoder.add_module('fc{}'.format(i), nn.Linear(self.hiddens_decoder[i-1], self.hiddens_decoder[i]))
            self.decoder.add_module('elu{}'.format(len(self.hiddens_decoder)), nn.ELU())
            self.decoder.add_module('fc{}'.format(len(self.hiddens_decoder)), nn.Linear(self.hiddens_decoder[-1], self.output_dim))
        else:
            self.decoder.add_module('fc{}'.format(1), nn.Linear(input_size, self.output_dim))

        input_size = self.z_dim
        self.hiddens_decoder = config['model']['generative']['hiddens_decoder']
        self.decoder_p = nn.Sequential()
        if -1 not in self.hiddens_decoder:
            for i in range(len(self.hiddens_decoder)):
                if i == 0:
                    self.decoder_p.add_module('fc{}'.format(i), nn.Linear(input_size, self.hiddens_decoder[i]))
                else:
                    self.decoder_p.add_module('elu{}'.format(i), nn.ELU())
                    self.decoder_p.add_module('fc{}'.format(i), nn.Linear(self.hiddens_decoder[i-1], self.hiddens_decoder[i]))
            self.decoder_p.add_module('elu{}'.format(len(self.hiddens_decoder)), nn.ELU())
            self.decoder_p.add_module('fc{}'.format(len(self.hiddens_decoder)), nn.Linear(self.hiddens_decoder[-1], self.output_dim * 2))
        else:
            self.decoder_p.add_module('fc{}'.format(1), nn.Linear(input_size, self.output_dim * 2))

        input_size = self.z_dim + self.treatment_hidden_dim
        self.hiddens_decoder = config['model']['generative']['hiddens_decoder']
        self.decoder_pa = nn.Sequential()
        if -1 not in self.hiddens_decoder:
            for i in range(len(self.hiddens_decoder)):
                if i == 0:
                    self.decoder_pa.add_module('fc{}'.format(i), nn.Linear(input_size, self.hiddens_decoder[i]))
                else:
                    self.decoder_pa.add_module('elu{}'.format(i), nn.ELU())
                    self.decoder_pa.add_module('fc{}'.format(i), nn.Linear(self.hiddens_decoder[i-1], self.hiddens_decoder[i]))
            self.decoder_pa.add_module('elu{}'.format(len(self.hiddens_decoder)), nn.ELU())
            self.decoder_pa.add_module('fc{}'.format(len(self.hiddens_decoder)), nn.Linear(self.hiddens_decoder[-1], self.output_dim * 2))
        else:
            self.decoder_pa.add_module('fc{}'.format(1), nn.Linear(input_size, self.output_dim * 2))

        if self.predict_X:
            input_size = self.z_dim
            self.hiddens_decoder = config['model']['generative']['hiddens_decoder']
            self.decoder_x = nn.Sequential()
            if -1 not in self.hiddens_decoder:
                for i in range(len(self.hiddens_decoder)):
                    if i == 0:
                        self.decoder_x.add_module('fc{}'.format(i), nn.Linear(input_size, self.hiddens_decoder[i]))
                    else:
                        self.decoder_x.add_module('elu{}'.format(i), nn.ELU())
                        self.decoder_x.add_module('fc{}'.format(i), nn.Linear(self.hiddens_decoder[i-1], self.hiddens_decoder[i]))
                self.decoder_x.add_module('elu{}'.format(len(self.hiddens_decoder)), nn.ELU())
                self.decoder_x.add_module('fc{}'.format(len(self.hiddens_decoder)), nn.Linear(self.hiddens_decoder[-1], self.input_size))
            else:
                self.decoder_x.add_module('fc{}'.format(1), nn.Linear(input_size, self.input_size))

            input_size = self.z_dim
            self.hiddens_decoder = config['model']['generative']['hiddens_decoder']
            self.decoder_p_x = nn.Sequential()
            if -1 not in self.hiddens_decoder:
                for i in range(len(self.hiddens_decoder)):
                    if i == 0:
                        self.decoder_p_x.add_module('fc{}'.format(i), nn.Linear(input_size, self.hiddens_decoder[i]))
                    else:
                        self.decoder_p_x.add_module('elu{}'.format(i), nn.ELU())
                        self.decoder_p_x.add_module('fc{}'.format(i), nn.Linear(self.hiddens_decoder[i-1], self.hiddens_decoder[i]))
                self.decoder_p_x.add_module('elu{}'.format(len(self.hiddens_decoder)), nn.ELU())
                self.decoder_p_x.add_module('fc{}'.format(len(self.hiddens_decoder)), nn.Linear(self.hiddens_decoder[-1], self.input_size * 2))
            else:
                self.decoder_p_x.add_module('fc{}'.format(1), nn.Linear(input_size, self.input_size * 2))

        self.epsilon = 1e-1

        input_dim = self.z_dim
        hidden_dim = self.hidden_dim
        output_dim = self.hidden_dim


    def init_hidden(self, batch_size):
        """Initialize LSTM Stealth"""
        device = next(self.parameters()).device
        self.hidden_state = torch.zeros(self.num_layers, batch_size, self.hidden_dim).to(device)
        self.cell_state = torch.zeros(self.num_layers, batch_size, self.hidden_dim).to(device)

    def build_H_t(self, H_t):
        if self.static_size > 0:
            if self.predict_X:
                x = H_t['vitals']
                x = torch.cat((x, H_t['static_features']), dim=-1)
            # when we don't predict x, we use static features as the current_covariates
            else:
                x = H_t['static_features']
        # if we use autoregressive, we need to use the previous output as the input
        if self.autoregressive:
            prev_outputs = H_t['prev_outputs']
            x = torch.cat((x, prev_outputs), dim=-1)
        
        previous_treatments = H_t['prev_treatments']
        x = torch.cat((x, previous_treatments), dim=-1) # (batch_size, seq_length, input_size)
        return x

    def init_hidden_history(self, H_t):
        x = self.build_H_t(H_t)
        Z_t, _ = self.lstm_history(x) # (batch_size, history_length, hidden_dim)
        # treatments = H_t['current_treatments']
        # y_hat = self.predict_y_history_net(torch.cat((Z_t, treatments), dim=-1))
        # loss = nn.MSELoss()(y_hat, H_t['outputs'])
        return Z_t[:, -1, :]

    def build_reverse_action_encoding(self, a_seq):
        """
        a_seq: (batch_size, tau, treatment_size)
        """
        # reverse the action sequence
        reversed_a_seq = torch.flip(a_seq, [1])
        # reversed_a_seq = a_seq
        outs, (_, _) = self.reverse_action_encoder(reversed_a_seq) # (batch_size, tau, treatment_hidden_dim)
        # reverse the output
        outs = torch.flip(outs, [1])
        return outs

    def build_hidden_states_v3(self, Z, a_seq):
        hidden_states = []
        for i in range(a_seq.size(1)):
            out = self.hidden_lstm(torch.cat((Z.unsqueeze(1), a_seq[:, i, :].unsqueeze(1)), dim=-1))
            Z = out[0][:, -1, :]
            hidden_states.append(Z)

        return torch.stack(hidden_states, dim=1)
    
    def build_hidden_states_v2(self, Z, a_seq):
        Z_expanded = Z.unsqueeze(1).repeat(1, a_seq.size(1), 1)  # [batch_size, seq_len, Z_dim]
        
        lstm_input = torch.cat((Z_expanded, a_seq), dim=-1)  # [batch_size, seq_len, Z_dim + action_dim]
        
        out, _ = self.hidden_lstm(lstm_input)  # out: [batch_size, seq_len, hidden_dim]
        
        hidden_states = out  # [batch_size, seq_len, hidden_dim]
        return hidden_states

    def build_hidden_states(self, Z, a_seq):
        hidden_states = []
        h_t, c_t = None, None  
        seq_len = a_seq.size(1)

        for i in range(seq_len):
            input_t = torch.cat((Z.unsqueeze(1), a_seq[:, i, :].unsqueeze(1)), dim=-1)  # [batch_size, 1, Z_dim + action_dim]
            out, (h_t, c_t) = self.hidden_lstm(input_t, (h_t, c_t)) if h_t is not None else self.hidden_lstm(input_t)
            Z = out[:, -1, :]  
            hidden_states.append(Z)

        hidden_states = torch.stack(hidden_states, dim=1)  # [batch_size, seq_len, hidden_dim]
        return hidden_states

    def build_reverse_hidden_states(self, hidden_states):
        reversed_hidden_states = torch.flip(hidden_states, [1])
        outs, (_, _) = self.hidden_lstm_reverse(reversed_hidden_states)
        outs = torch.flip(outs, [1])
        return outs


    def build_action_encoding(self, a_seq):
        """
        a_seq: (batch_size, tau, treatment_size)
        """
        outs, (_, _) = self.action_encoder(a_seq.detach()) # (batch_size, tau, treatment_hidden_dim)
        # outs, (_, _) = self.action_encoder(a_seq)
        
        return outs

    def predict_actions(self, Z_init, a_seq):
        a_seq = a_seq[:, :-1, :]
        # append the initial zero hidden state
        a_seq = torch.cat((torch.zeros(a_seq.size(0), 1, a_seq.size(-1)).to(a_seq.device), a_seq), dim=1) # (batch_size, tau, treatment_size)
        a_seq, (_, _) = self.action_encoder(a_seq) # (batch_size, tau, treatment_size)
        Z_init_expanded = Z_init.unsqueeze(1).repeat(1, a_seq.size(1), 1)
        inputs = torch.cat((Z_init_expanded, a_seq), dim=-1) # (batch_size, tau, z_dim + treatment_size)
        action_hat = self.action_decoder(inputs) # (batch_size, tau, treatment_size)
        return action_hat

    def loss_predict_actions(self, Z_init, a_seq):
        action_hat = self.predict_actions(Z_init, a_seq)
        loss = nn.MSELoss(reduction='none')(action_hat, a_seq)
        
        return batch_mean_except_first(loss)

    def loss_predict_actions_beta(self, Z_init, a_seq):
        a_seq_ori = a_seq.clone()
        a_seq = a_seq[:, :-1, :]
        a_seq = torch.cat((torch.zeros(a_seq.size(0), 1, a_seq.size(-1)).to(a_seq.device), a_seq), dim=1) # (batch_size, tau, treatment_size)
        a_seq, (_, _) = self.action_encoder(a_seq) # (batch_size, tau, treatment_size)
        
        Z_init_expanded = Z_init.unsqueeze(1).repeat(1, a_seq.size(1), 1)
        # print(f'a_seq: {a_seq.size()}, Z_init_expanded: {Z_init_expanded.size()}, Z_init: {Z_init.size()}')
        inputs = torch.cat((Z_init_expanded, a_seq), dim=-1) # (batch_size, tau, z_dim + treatment_size)
        out = self.action_decoder_beta(inputs) # (batch_size, tau, treatment_size)

        alpha, beta = out[:, :, :self.treatment_size], out[:, :, self.treatment_size:]
        alpha = F.softplus(alpha) + self.epsilon
        beta = F.softplus(beta) + self.epsilon
        
        dist_beta = dist.Beta(alpha, beta)
        entropy = dist_beta.entropy()

        log_likelihood = dist_beta.log_prob(a_seq_ori.clamp(self.epsilon, 1 - self.epsilon)) 
        loss = self.loss_predict_actions(Z_init, a_seq_ori)

        # return -log_likelihood.sum(dim=(-1, -2)).mean() - entropy * self.config.model.generative.entropy_lambda + loss

        return -batch_mean_except_first(log_likelihood) - batch_mean_except_first(entropy) * self.config.model.generative.entropy_lambda + loss

    def predict_actions_binary(self, Z_init, a_seq):
        a_seq = a_seq[:, :-1, :]
        # append the initial zero hidden state
        a_seq = torch.cat((torch.zeros(a_seq.size(0), 1, a_seq.size(-1)).to(a_seq.device), a_seq), dim=1) # (batch_size, tau, treatment_size)
        a_seq, (_, _) = self.action_encoder(a_seq) # (batch_size, tau, treatment_size)
        Z_init_expanded = Z_init.unsqueeze(1).repeat(1, a_seq.size(1), 1)
        inputs = torch.cat((Z_init_expanded, a_seq), dim=-1) # (batch_size, tau, z_dim + treatment_size)
        
        logits = self.action_decoder(inputs) # (batch_size, tau, treatment_size)
        
        action_prob = torch.sigmoid(logits)
        action_prob = action_prob.clamp(self.epsilon, 1 - self.epsilon)  #Numerical stability
        
        return action_prob

    def loss_predict_actions_binary(self, Z_init, a_seq):
        action_prob = self.predict_actions_binary(Z_init, a_seq)
    
        loss = nn.BCELoss(reduction=None)(action_prob, a_seq)
        return batch_mean_except_first(loss)

    def loss_predict_actions_bern(self, Z_init, a_seq):
        a_seq_ori = a_seq
        a_seq = a_seq[:, :-1, :]
        a_seq = torch.cat((torch.zeros(a_seq.size(0), 1, a_seq.size(-1)).to(a_seq.device), a_seq), dim=1) # (batch_size, tau, treatment_size)
        a_seq, (_, _) = self.action_encoder(a_seq) # (batch_size, tau, treatment_size)
        a_seq = a_seq.clamp(self.epsilon, 1 - self.epsilon)
        Z_init_expanded = Z_init.unsqueeze(1).repeat(1, a_seq.size(1), 1)
        # print(f'a_seq: {a_seq.size()}, Z_init_expanded: {Z_init_expanded.size()}, Z_init: {Z_init.size()}')
        inputs = torch.cat((Z_init_expanded, a_seq), dim=-1) # (batch_size, tau, z_dim + treatment_size)
        out = self.action_decoder_beta(inputs) # (batch_size, tau, treatment_size)
        mu, logvar = out[:, :, :self.treatment_size], out[:, :, self.treatment_size:]

        probs = torch.sigmoid(mu)
        probs = probs.clamp(self.epsilon, 1 - self.epsilon)  #Numerical stability
        # print(f"shape of beta {beta.shape}")
        dist_bern = torch.distributions.Bernoulli(probs=probs)
        entropy = dist_bern.entropy()

        log_likelihood = dist_bern.log_prob(a_seq_ori) 
        loss = self.loss_predict_actions_binary(Z_init, a_seq_ori)

        return -batch_mean_except_first(log_likelihood) - batch_mean_except_first(entropy) * self.config.model.generative.entropy_lambda + loss


    def decode_action_step(self, Z_s):
        """
        Z_s: (batch_size, z_dim)
        """
        a_s = self.action_decoder_step(Z_s)
        # a_s = torch.sigmoid(a_s)
        # a_s = torch.clamp(a_s, min=0, max=1)
        mu, logvar = a_s[:, :self.treatment_size], a_s[:, self.treatment_size:]

        return mu, logvar

    def forward(self, Z_s_prev, a_s):
        """
        Z_s_prev: (batch_size, z_dim)
        a_s: (batch_size, treatment_dim)
        """
        #Check and Initialize Stealth
        batch_size = Z_s_prev.size(0)
        if self.hidden_state is None:
            self.init_hidden(batch_size)
        
        #Prepare LSTM input
        input = torch.cat([Z_s_prev, a_s], dim=-1).unsqueeze(1)  #Add timestep dimension
        
        #LSTM Forward Propagation
        lstm_out, (self.hidden_state, self.cell_state) = self.lstm(
            input, (self.hidden_state, self.cell_state)
        )
        lstm_out = lstm_out.squeeze(1)  #Remove timestep dimension
        lstm_out = torch.cat([lstm_out, a_s], dim=-1)

        
        #Calculate prior distribution parameters
        p_mu = self.fc_mu(lstm_out)
        p_logvar = self.fc_logvar(lstm_out)
        # p_logvar = F.softplus(p_logvar)
        
        return p_mu, p_logvar
    
    def decode(self, Z_s):
        return self.decoder(Z_s)

    def decode_p(self, Z_s):
        out = self.decoder_p(Z_s)
        mu, p_logvar = out[..., :self.output_dim], out[..., self.output_dim:]
        # print(f"p_std : {p_logvar.mean()}, p_std : {p_logvar.shape}")
        
        return mu, p_logvar

    def decode_p_a(self, Z_s, a):
        # print(Z_s.shape, a.shape)
        input = torch.cat((Z_s, a), dim=-1)
        out = self.decoder_pa(input)
        mu, p_logvar = out[..., :self.output_dim], out[..., self.output_dim:]
        # print(f"p_std : {p_logvar.mean()}, p_std : {p_logvar.shape}")
        
        return mu, p_logvar

    def decoding_Y_loss(self, Z, y):
        mu, p_logvar = self.decode_p(Z)
        p_std = torch.exp(0.5 * p_logvar) + 1e-6
        normal_dist = Normal(mu, p_std)
        # print(f"mu shape:{mu.shape}")
        # print(f"y shape:{y.shape}")
        p = normal_dist.log_prob(y).sum(dim=-1).mean()
        # print(f"p_std : {p_std.mean()}")

        loss = F.mse_loss(mu, y, reduction='none')
        return batch_mean_except_first(loss) * 0.1
        return -p

    def decoding_Y_loss_2(self, Z, y, a):
        mu, p_logvar = self.decode_p_a(Z, a)
        p_std = torch.exp(0.5 * p_logvar) + 1e-6
        p_std = torch.ones(p_std.shape, device=p_std.device) * 0.1
        normal_dist = Normal(mu, p_std)
        # print(f"mu shape:{mu.shape}")
        # print(f"y shape:{y.shape}")
        p = normal_dist.log_prob(y).sum(dim=-1).mean()
        # print(f"p_std : {p_std.mean()}")

        loss = F.mse_loss(mu, y, reduction='none')
        return batch_mean_except_first(loss)
    
    def decode_x(self, Z_s):
        return self.decoder(Z_s)

    def decode_p_x(self, Z_s):
        out = self.decoder_p_x(Z_s)
        mu, p_logvar = out[..., :self.input_size], out[..., self.input_size:]
        # print(f"p_std : {p_logvar.mean()}, p_std : {p_logvar.shape}")
        
        return mu, p_logvar
    
    def decoding_X_loss(self, Z, x):
        mu, p_logvar = self.decode_p_x(Z)
        p_std = torch.exp(0.5 * p_logvar) + 1e-6
        normal_dist = Normal(mu, p_std)
        # print(f"mu shape:{mu.shape}")
        # print(f"y shape:{y.shape}")
        # p = normal_dist.log_prob(x).sum(dim=-1).mean()
        # print(f"p_std : {p_std.mean()}")
        loss = F.mse_loss(mu, x, reduction='none')
        return batch_mean_except_first(loss)


    def beta_loss(self, Z_s, a_s):
        a_s = a_s.clamp(self.epsilon, 1 - self.epsilon)
        alpha, beta = self.decode_action_step(Z_s)
        # alpha = F.softplus(mu) + self.epsilon  #  alpha > 0
        # beta = F.softplus(logvar) + self.epsilon    # beta > 0
        alpha = F.softplus(alpha) 
        beta = F.softplus(beta)
        
        dist_beta = dist.Beta(alpha, beta)
        log_likelihood = dist_beta.log_prob(a_s)
        
        entropy_reg = self.config.exp.entropy_reg
        entropy = dist_beta.entropy()
        
        if self.beta_bound:
            loss_reg = log_likelihood.clamp(self.beta_bound) * 0.1
        else:
            loss_reg = log_likelihood * 0.1
        loss = loss_reg - entropy_reg * entropy
        return batch_mean_except_first(loss)

    def bern_loss(self, Z_s, a_s):
        logits, _ = self.decode_action_step(Z_s)
        probs = torch.sigmoid(logits)
        
        dist_bern = torch.distributions.Bernoulli(probs=probs)
        log_likelihood = dist_bern.log_prob(a_s)

        entropy_reg = 1
        entropy = dist_bern.entropy()
        
        #Calculate loss
        if self.beta_bound:
            loss_reg = log_likelihood.clamp(self.beta_bound) * 0.1
        else:
            loss_reg = log_likelihood * 0.1
            
        loss = -loss_reg - entropy_reg * entropy
        
        return batch_mean_except_first(loss)

    def bce_loss(self, Z_s, a_s):
        a_pred, _ = self.decode_action_step(Z_s)
        a_pred = torch.sigmoid(a_pred)
        bce_loss = -F.binary_cross_entropy(a_pred, a_s, reduction='none')
        return batch_mean_except_first(bce_loss)

    def mse_loss(self, Z_s, a_s):
        a_pred, _ = self.decode_action_step(Z_s)
        a_pred = torch.sigmoid(a_pred)
        mse_loss = F.mse_loss(a_pred, a_s, reduction='none')
        return -batch_mean_except_first(mse_loss)
  
    def reset_states(self):
        """Reset LSTM Stealth"""
        self.hidden_state = None
        self.cell_state = None

    def reparameterize(self, mu, logvar):
        std = torch.exp(0.5 * logvar)
        eps = torch.randn_like(std)
        return mu + eps * std