

import numpy as np
import torch
import torch.nn as nn


class SpectralConv1d(nn.Module):
    def __init__(self, in_channels, out_channels, modes1):
        super(SpectralConv1d, self).__init__()
        """
        1D Fourier layer. It does FFT, linear transform, and Inverse FFT.    
        """

        self.in_channels = in_channels
        self.out_channels = out_channels
        self.modes1 = modes1  # Number of Fourier modes to multiply, at most floor(N/2) + 1

        self.scale = (1 / (in_channels * out_channels))
        self.weights1 = nn.Parameter(
            self.scale * torch.rand(in_channels, out_channels, self.modes1, dtype=torch.cdouble))

    # Complex multiplication
    def compl_mul1d(self, input, weights):
        # (batch, in_channel, x ), (in_channel, out_channel, x) -> (batch, out_channel, x)
        return torch.einsum("bix,iox->box", input, weights)

    def forward(self, x):
        batchsize = x.shape[0]
        # Compute Fourier coeffcients up to factor of e^(- something constant)
        x_ft = torch.fft.rfft(x)

        # Multiply relevant Fourier modes
        out_ft = torch.zeros(batchsize, self.out_channels, x.size(-1) // 2 + 1, device=x.device, dtype=torch.cdouble)
        out_ft[:, :, :self.modes1] = self.compl_mul1d(x_ft[:, :, :self.modes1], self.weights1)

        # Return to physical space
        x = torch.fft.irfft(out_ft, n=x.size(-1))
        return x


class ConditionalFourierNeuralOperator(torch.nn.Module):
    def __init__(self, modes, width, d_cond=0, max_t=1000, activation='gelu'):
        super(ConditionalFourierNeuralOperator, self).__init__()

        """
        Fourier Neural Operator with time conditioning.
        Time conditioning is handled by appending t as an input node.
        Assumes all functions are observed on a uniform gridding of the input space. 
        
        Comments from original implementation:
        The overall network. It contains 4 layers of the Fourier layer.
        1. Lift the input to the desire channel dimension by self.fc0 .
        2. 4 layers of the integral operators u' = (W + K)(u).
            W defined by self.w; K defined by self.conv .
        3. Project from the channel space to the output space by self.fc1 and self.fc2 .

        input: the solution of the initial condition and location (a(x), x)
        input shape: (batchsize, x=s, c=2)
        output: the solution of a later timestep
        output shape: (batchsize, x=s, c=1)
        """

        if activation == 'gelu':
            self.activation = nn.GELU()
        elif activation == 'tanh':
            self.activation = nn.Tanh()
        else:
            raise NotImplementedError(f'Activation {activation} not supported (yet)')

        self.modes = modes
        self.width = width
        self.max_t = max_t
        self.padding = 2  # pad the domain if input is non-periodic

        self.fc0 = nn.Linear(3 + d_cond, self.width)  # Input is (u(x), x, t, z) where z has dim d_cond
        self.conv0 = SpectralConv1d(self.width, self.width, self.modes)
        self.conv1 = SpectralConv1d(self.width, self.width, self.modes)
        self.conv2 = SpectralConv1d(self.width, self.width, self.modes)
        self.conv3 = SpectralConv1d(self.width, self.width, self.modes)
        self.w0 = nn.Conv1d(self.width, self.width, 1)
        self.w1 = nn.Conv1d(self.width, self.width, 1)
        self.w2 = nn.Conv1d(self.width, self.width, 1)
        self.w3 = nn.Conv1d(self.width, self.width, 1)

        self.fc1 = nn.Linear(self.width, 128)
        self.fc2 = nn.Linear(128, 1)

    def forward(self, t, u, z):
        # u: function values observed on uniform grid -- (batch_size, n_x, d_u)
        # t: time -- (batch_size, )
        # z: conditioning information -- (batch_size, n_cond)  where n_cond is dimensionality of conditioning
        # -- e.g. conditioning on a start/end value of a function, n_cond=2

        #  clean up
        if u.dim() == 2:
            u = u.unsqueeze(-1)
        if t.dim() == 0:
            t = torch.ones(u.shape[0], device=t.device) * t
            
        assert u.dim() == 3, f'Got shape {u.shape}'
        assert t.dim() == 1
        assert t.shape[0] == u.shape[0]
        assert z.shape[0] == u.shape[0]

        # Concatenate x values (on grid) and time to input
        x_grid = self.get_grid(u.shape, u.device)
        t = t / self.max_t
        t = t.unsqueeze(-1)
        #t = t.reshape(-1, 1).repeat(1, u.shape[1]).unsqueeze(-1)
        # Concatenate time and conditioning information -- which is then appended to every row of input
        tz = torch.hstack((t, z)).repeat(u.shape[1], 1, 1).permute(1, 0, 2)
        x = torch.cat((u, x_grid, tz), dim=-1)  # (batch_size, n_x, d_u + n_cond + 1) tensor containing (u(x), x, t, z)
  
        x = x.double() #   clean up

        x = self.fc0(x)
        x = x.permute(0, 2, 1)
        # x = F.pad(x, [0,self.padding]) # pad the domain if input is non-periodic

        x1 = self.conv0(x)
        x2 = self.w0(x)
        x = x1 + x2
        x = self.activation(x)

        x1 = self.conv1(x)
        x2 = self.w1(x)
        x = x1 + x2
        x = self.activation(x)

        x1 = self.conv2(x)
        x2 = self.w2(x)
        x = x1 + x2
        x = self.activation(x)

        x1 = self.conv3(x)
        x2 = self.w3(x)
        x = x1 + x2

        # x = x[..., :-self.padding] # pad the domain if input is non-periodic
        x = x.permute(0, 2, 1)
        x = self.fc1(x)
        x = self.activation(x)
        x = self.fc2(x)
        #return x
        return x.squeeze()  #   clean up

    def get_grid(self, shape, device):
        batchsize, size_x = shape[0], shape[1]
        gridx = torch.tensor(np.linspace(0, 1, size_x), dtype=torch.float, device=device)
        gridx = gridx.reshape(1, size_x, 1).repeat([batchsize, 1, 1])
        return gridx
