import abc
from pathlib import Path
from typing import List, Optional, Callable, Mapping

import torch
import torchaudio
import tqdm
from math import sqrt, ceil

from audio_diffusion_pytorch.diffusion import Schedule
from torch.utils.data import DataLoader

from main.data import assert_is_audio, SeparationDataset
from main.module_base import Model

class Separator(torch.nn.Module, abc.ABC):
    def __init__(self):
        super().__init__()
        
    @abc.abstractmethod
    def separate(mixture, num_steps) -> Mapping[str, torch.Tensor]:
        ...
    
    
class Weakly_MSDMSeparator_opt(Separator):
    def __init__(self, stem_to_model, sigma_schedule: Schedule, **kwargs):
        super().__init__()
        self.stem_to_model = stem_to_model
        self.stems = self.stem_to_model.keys()
        self.sigma_schedule = sigma_schedule
        self.separation_kwargs = kwargs
    
    def separate(self, mixture: torch.Tensor, num_steps:int = 100):
        device = self.stem_to_model['guitar'].device
        mixture = mixture.to(device)
        batch_size, _, length_samples = mixture.shape
        fns = [m.model.diffusion.denoise_fn for m in self.stem_to_model.values()]

        def denoise_fn(x, sigma):
            xs = [x[:, i:i+1] for i in range(4)]
            xs = [fn(x,sigma=sigma) for fn,x in zip(fns, xs)]
            return torch.cat(xs, dim=1)

        y = separate_mixture_opt(
            mixture=mixture,
            denoise_fn=denoise_fn,
            sigmas=self.sigma_schedule(num_steps, device),
            noises=torch.randn(batch_size, len(self.stems), length_samples).to(device),
            **self.separation_kwargs,
        )
        return {stem:y[:,i:i+1,:] for i,stem in enumerate(self.stems)}


def differential_with_dirac(x, sigma, denoise_fn, mixture, source_id=0):
    num_sources = x.shape[1]
    x[:, [source_id], :] = mixture - (x.sum(dim=1, keepdim=True) - x[:, [source_id], :])
    score = (x - denoise_fn(x, sigma=sigma)) / sigma
    scores = [score[:, si] for si in range(num_sources)]
    ds = [s - score[:, source_id] for s in scores]
    return torch.stack(ds, dim=1)

@torch.no_grad()
def separate_mixture_opt(
    mixture: torch.Tensor, 
    denoise_fn: Callable,
    sigmas: torch.Tensor,
    noises: Optional[torch.Tensor],
    differential_fn: Callable = differential_with_dirac,
    s_churn: float = 0.0, # > 0 to add randomness
    num_resamples: int = 1,
    use_tqdm: bool = False,
):      
    def prox(x, mixture):
        prox_x = x + 0.25 * (mixture - torch.sum(x, dim=1, keepdim=True)).repeat(1,4,1)
        return prox_x
    # Set initial noise
    x = sigmas[0] * noises # [batch_size, num-sources, sample-length]
    source_id = 0
    vis_wrapper  = tqdm.tqdm if use_tqdm else lambda x:x 
    # print(sigmas)
    x_0 = mixture.repeat(1,4,1)/4
    momentum = None
    lr = 0.1
    n = 5
    beta=0.5
    for i in vis_wrapper(range(len(sigmas) - 1)):
        # if i == 100:
        #     lr *= 0.1
        # print(lamb * torch.eye(4).cuda() + torch.matmul(A.T, A))
        sigma, sigma_next = sigmas[i], sigmas[i+1]
        for k in range(n):
            A = torch.ones([1, 4]).cuda()
            # B = torch.linalg.inv(torch.eye(4).cuda() + lamb * torch.matmul(A.T, A))
            num_sources = x.shape[1]
            if x_0 is None:
                # score = (x - denoise_fn(x, sigma=sigma)) / sigma # score_function
                # ds = score
                # x_0_pred = 1/(sqrt(1-sigma**2)) * (x - sigma * ds)
                # x_0 = x_0_pred # 初始化x_0
                x_0_pred = denoise_fn(x, sigma=sigma)
                x_0 = denoise_fn(torch.randn_like(x), sigma=sigma)
            else:
                x = x_0 + sigma * torch.randn_like(x_0)
                # test moving average
                # x = sqrt(sigma) * x_0 + sqrt(1-(1-sqrt(sigma))**2) * sigma * torch.randn_like(x_0) + (1-sqrt(sigma)) * x
                # score = (x - denoise_fn(x, sigma=sigma)) / sigma # score_function
                # ds = score
                # x_0_pred = 1/(sqrt(1-sigma**2)) * (x - sigma * ds)
                x_0_pred = denoise_fn(x, sigma=sigma)

            diff = (x_0_pred - x_0) - (torch.sum(x_0, dim=1, keepdim=True) - mixture).repeat(1,4,1)
            if momentum is None:
                momentum=diff
            else:
                momentum = beta * momentum + (1-beta) * diff
            x_0 += lr * momentum
            # x_0 = prox(x_0, mixture)
            loss_obs = torch.mean(torch.norm(torch.sum(x_0, dim=1, keepdim=True) - mixture, dim=2)).item()
            loss_cons = torch.mean(torch.norm(x_0-x_0_pred, dim=[1,2])).item()
            print('obs loss:{}, cons loss:{}, lr:{}, sigma:{}'.format(loss_obs, loss_cons, lr, sigma))
        # 更新x
        # score = (x - denoise_fn(x, sigma=sigma)) / sigma # score_function
        # x = x + score * (sigma_next - sigma)
    # diff = -(torch.sum(x_0, dim=1, keepdim=True) - mixture).repeat(1,4,1)
    # x_0 += 0.25 * diff
    return x_0.cpu().detach()
    # return x.cpu().detach()