import torch
import torch.nn as nn

class RecurrentCycle(torch.nn.Module):

    def __init__(self, cycle_len, channel_size):
        super(RecurrentCycle, self).__init__()
        self.cycle_len = cycle_len
        self.channel_size = channel_size
        self.data = torch.nn.Parameter(torch.zeros(cycle_len, channel_size), requires_grad=True)

    def forward(self, index, length):

        gather_index = (index.view(-1, 1) + torch.arange(length, device=index.device).view(1, -1)) % self.cycle_len

        return self.data[gather_index]

class SeriesDecomp(nn.Module):
    def __init__(self, kernel_size):
        super().__init__()
        assert kernel_size % 2 == 1
        self.avg_pool = nn.AvgPool1d(
            kernel_size=kernel_size,
            stride=1,
            padding=(kernel_size - 1) // 2
        )

    def forward(self, x):

        x_perm = x.permute(0, 2, 1)

        trend = self.avg_pool(x_perm)

        trend = trend.permute(0, 2, 1)

        residual = x - trend

        return residual, trend

class Model(nn.Module):
    def __init__(self, configs):
        super(Model, self).__init__()

        self.seq_len = configs.seq_len
        self.pred_len = configs.pred_len
        self.enc_in = configs.enc_in
        self.cycle_len = configs.cycle
        self.model_type = configs.model_type
        self.d_model = configs.d_model
        self.use_revin = configs.use_revin
        self.n = configs.n

        self.cycleQueue = RecurrentCycle(cycle_len=self.cycle_len, channel_size=self.enc_in)

        self.delta = nn.Parameter(0.02 * torch.randn(1, self.enc_in))
        self.Multi_Expert = nn.ModuleList([nn.Sequential(nn.Linear(self.seq_len, self.d_model),
                                    nn.ReLU(),
                                    nn.Linear(self.d_model, self.pred_len)) for i in range(self.n)])

        assert self.model_type in ['linear', 'mlp']
        if self.model_type == 'linear':
            self.model = nn.Linear(self.seq_len, self.pred_len)

        elif self.model_type == 'mlp':
            self.model = nn.Sequential(
                nn.Linear(self.seq_len, self.d_model),
                nn.ReLU(),
                nn.Linear(self.d_model, self.pred_len),
            )

        self.decom = SeriesDecomp(kernel_size=3)


    def forward(self, x, cycle_index):

        if self.use_revin:
            seq_mean = torch.mean(x, dim=1, keepdim=True)
            seq_var = torch.var(x, dim=1, keepdim=True) + 1e-5
            x = (x - seq_mean) / torch.sqrt(seq_var)

        k = x
        list = []

        for i in range(self.n):
            r,_ = self.decom(k)
            list.append(r)
            k = r
        for rr in list:
            x -= rr

        x = x - self.cycleQueue(cycle_index, self.seq_len)

        y = self.Multi_Expert(x.permute(0, 2, 1)).permute(0, 2, 1)

        y = y + self.cycleQueue((cycle_index + self.seq_len) % self.cycle_len, self.pred_len)

        out_r_list = []

        for j in range(len(list)):

            out = self.modelr[j](list[j].permute(0,2,1)).permute(0,2,1)

            y+=out
            out_r_list.append(out)

        y += self.delta

        if self.use_revin:
            y = y * torch.sqrt(seq_var) + seq_mean

        return y
