import os
from os.path import join as ospj
import logging
import numpy as np

import torch
import torchvision
import torch.nn as nn
from torch.cuda.amp import GradScaler, autocast
from torch.utils.tensorboard import SummaryWriter
from tqdm import tqdm
from utils import accuracy, CheckpointIO, MultiDimAverageMeter

from data_aug.data_loader import get_original_loader, get_val_loader, InputFetcher
from training.SimCLRSolver import SimCLRSolver


class LinearEvalSolver(SimCLRSolver):
    def __init__(self, args):
        super(LinearEvalSolver, self).__init__(args)
        self.writer = SummaryWriter(ospj(args.log_dir, 'linear_eval'))

    def effective_rank(self, w):
        s = w.cpu().svd(compute_uv=False)[1]
        s_hat = s / s.sum()
        return - (s_hat * s_hat.log()).sum()

    def validation(self, fetcher):
        self.nets.encoder.eval()
        self.nets.classifier.eval()

        attrwise_acc_meter = MultiDimAverageMeter(self.attr_dims)

        total_correct, total_num = 0, 0

        for images, labels, bias, _ in tqdm(fetcher):
            label = labels.to(self.device)
            images = images.to(self.device)
            bias = bias.to(self.device)

            with torch.no_grad():
                aux = self.nets.encoder(images, penultimate=True)
                features = aux['penultimate']
                logit = self.nets.classifier(features)
                pred = logit.data.max(1, keepdim=True)[1].squeeze(1)
                correct = (pred == label).long()

                total_correct += correct.sum()
                total_num += correct.shape[0]

            attr = torch.cat((labels.view(-1,1).to(self.device), bias.view(-1,1).to(self.device)), dim=1)
            attrwise_acc_meter.add(correct.cpu(), attr.cpu())

        if self.args.data == 'stl10mnist':
            oversample_token = 'oversample' if self.args.mode == 'oversample' else 'normal'
            attrwise_acc_meter.save(ospj(self.args.log_dir, f'valid_acc_{self.args.bias_ratio}_{oversample_token}.pth'))
        else:
            attrwise_acc_meter.save(ospj(self.args.log_dir, f'valid_acc.pth'))

        print(attrwise_acc_meter.cum.view(self.attr_dims[0], -1))
        print(attrwise_acc_meter.cnt.view(self.attr_dims[0], -1))

        total_acc = total_correct / float(total_num)
        accs = attrwise_acc_meter.get_mean()

        self.nets.encoder.train()
        self.nets.classifier.train()
        return total_acc, accs

    def report_validation(self, valid_attrwise_acc, valid_acc, step=0):
        eye_tsr = torch.eye(self.attr_dims[0]).long()
        valid_acc_align = valid_attrwise_acc[eye_tsr == 1].mean().item()
        valid_acc_conflict = valid_attrwise_acc[eye_tsr == 0].mean().item()

        all_acc = dict()
        for acc, key in zip([valid_acc, valid_acc_align, valid_acc_conflict],
                            ['Acc/total', 'Acc/align', 'Acc/conflict']):
            all_acc[key] = acc
            self.writer.add_scalar(key, acc, global_step=step)
        log = f"(Validation) Iteration [{step}], "
        log += ' '.join(['%s: [%.4f]' % (key, value) for key, value in all_acc.items()])
        logging.info(log)
        print(log)

    def save_score_idx(self, loader):
        self.nets.encoder.eval()
        self.nets.classifier.eval()
        dataset = get_original_loader(self.args, return_dataset=True, simclr_aug=False)
        num_data = len(dataset)

        iterator = enumerate(loader)
        score_idx = torch.zeros(num_data).to(self.device)
        wrong_idx = torch.zeros(num_data).to(self.device)
        debias_idx = torch.zeros(num_data).to(self.device)
        total_num = 0

        for _, (images, labels, bias_labels, idx) in iterator:
            idx = idx.to(self.device)
            labels = labels.to(self.device)
            bias_labels = bias_labels.to(self.device)
            images= images.to(self.device)

            with torch.no_grad():
                aux = self.nets.encoder(images, freeze=True, penultimate=True)
                features_penul = aux['penultimate']
                logits = self.nets.classifier(features_penul)

                # bias score
                bias_prob = nn.Softmax()(logits)[torch.arange(logits.size(0)), labels]
                bias_score = 1 - bias_prob

                # wrong
                pred = logits.data.max(1, keepdim=True)[1].squeeze(1)
                wrong = (pred != labels).long()

                # true label
                debiased = (labels != bias_labels).float()

                for i, v in enumerate(idx):
                    score_idx[v] = bias_score[i]
                    debias_idx[v] = debiased[i]
                    wrong_idx[v] = wrong[i]

            total_num += labels.shape[0]

        if not self.args.finetune: assert total_num == len(score_idx)
        print(f'Average bias score: {score_idx.mean()}')

        self.nets.encoder.train()
        self.nets.classifier.train()
        score_idx_path = lambda x: ospj(self.args.checkpoint_dir, f'score_idx{x}.pth')
        wrong_idx_path = lambda x: ospj(self.args.checkpoint_dir, f'wrong_idx{x}.pth')
        debias_idx_path = lambda x: ospj(self.args.checkpoint_dir, f'debias_idx{x}.pth')

        if self.args.data == 'stl10mnist':
            score_idx_path = score_idx_path(f'_{self.args.bias_ratio}')
            wrong_idx_path = wrong_idx_path(f'_{self.args.bias_ratio}')
            debias_idx_path = debias_idx_path(f'_{self.args.bias_ratio}')
        else:
            score_idx_path = score_idx_path('')
            wrong_idx_path = wrong_idx_path('')
            debias_idx_path = debias_idx_path('')

        torch.save(score_idx, score_idx_path)
        torch.save(wrong_idx, wrong_idx_path)
        torch.save(debias_idx, debias_idx_path)
        print(f'Saved bias score in {score_idx_path}')
        self.pseudo_label_precision_recall(wrong_idx, debias_idx)

    def pseudo_label_precision_recall(self, wrong_label, debias_label):
        if self.args.data == 'celebA':
            debias_label = 1 - debias_label

        print(torch.sum(wrong_label))
        print(torch.sum(debias_label))

        spur_precision = torch.sum(
                (wrong_label == 1) & (debias_label == 1)
            ) / torch.sum(wrong_label)
        premsg = f"Spurious precision: {spur_precision}"
        print(premsg)
        logging.info(premsg)

        spur_recall = torch.sum(
                (wrong_label == 1) & (debias_label == 1)
            ) / torch.sum(debias_label)
        recmsg = f"Spurious recall: {spur_recall}"
        print(recmsg)
        logging.info(recmsg)

    def validate_imagenet(self, val_loader, num_classes=9, num_clusters=9,
                          num_cluster_repeat=3, key=None):
        self.nets.encoder.eval()
        self.nets.classifier.eval()

        total = 0
        f_correct = 0
        num_correct = [np.zeros([num_classes, num_clusters]) for _ in range(num_cluster_repeat)]
        num_instance = [np.zeros([num_classes, num_clusters]) for _ in range(num_cluster_repeat)]

        with torch.no_grad():
            for images, labels, bias_labels, index in val_loader:

                images, labels = images.to(self.device), labels.to(self.device)
                for bias_label in bias_labels:
                    bias_label.to(self.device)

                aux = self.nets.encoder(images, penultimate=True)
                features = aux['penultimate']
                output = self.nets.classifier(features)

                batch_size = labels.size(0)
                total += batch_size

                if key == 'unbiased':
                    num_correct, num_instance = self.imagenet_unbiased_accuracy(
                        output.data, labels, bias_labels,
                        num_correct, num_instance, num_cluster_repeat)
                else:
                    f_correct += self.n_correct(output, labels)

        self.nets.encoder.train()
        self.nets.classifier.train()

        if key == 'unbiased':
            result = {'num_correct': np.array(num_correct),
                      'num_instance': np.array(num_instance)}
            np.save(ospj(self.args.log_dir, 'unbiased_acc_array.npy'), result)
            for k in range(num_cluster_repeat):
                x, y = [], []
                _num_correct, _num_instance = num_correct[k].flatten(), num_instance[k].flatten()
                for i in range(_num_correct.shape[0]):
                    __num_correct, __num_instance = _num_correct[i], _num_instance[i]
                    if __num_instance >= 10:
                        x.append(__num_instance)
                        y.append(__num_correct / __num_instance)
                f_correct += sum(y) / len(x)

            return f_correct / num_cluster_repeat
        else:
            return f_correct / total

    def imagenet_save_score_idx(self, loader):
        self.nets.encoder.eval()
        self.nets.classifier.eval()
        dataset = get_original_loader(self.args, return_dataset=True, simclr_aug=False)
        num_data = len(dataset)

        iterator = enumerate(loader)
        score_idx = torch.zeros(num_data).to(self.device)
        wrong_idx = torch.zeros(num_data).to(self.device)
        total_num = 0

        for _, (images, labels, _, idx) in iterator:
            idx = idx.to(self.device)
            labels = labels.to(self.device)
            images= images.to(self.device)

            with torch.no_grad():
                aux = self.nets.encoder(images, freeze=True, penultimate=True)
                features_penul = aux['penultimate']
                logits = self.nets.classifier(features_penul)

                # bias score
                bias_prob = nn.Softmax()(logits)[torch.arange(logits.size(0)), labels]
                bias_score = 1 - bias_prob

                # wrong
                pred = logits.data.max(1, keepdim=True)[1].squeeze(1)
                wrong = (pred != labels).long()

                for i, v in enumerate(idx):
                    score_idx[v] = bias_score[i]
                    wrong_idx[v] = wrong[i]

            total_num += labels.shape[0]

        assert total_num == len(score_idx)
        print(f'Average bias score: {score_idx.mean()}')

        self.nets.encoder.train()
        self.nets.classifier.train()
        score_idx_path = ospj(self.args.checkpoint_dir, f'score_idx.pth')
        wrong_idx_path = ospj(self.args.checkpoint_dir, f'wrong_idx.pth')

        torch.save(score_idx, score_idx_path)
        torch.save(wrong_idx, wrong_idx_path)
        print(f'Saved bias score in {score_idx_path}')

    def imagenet_unbiased_accuracy(self, outputs, labels, cluster_labels,
                                   num_correct, num_instance,
                                   num_cluster_repeat=3):
        for j in range(num_cluster_repeat):
            for i in range(outputs.size(0)):
                output = outputs[i]
                label = labels[i]
                cluster_label = cluster_labels[j][i]

                _, pred = output.topk(1, 0, largest=True, sorted=True)
                correct = pred.eq(label).view(-1).float()

                num_correct[j][label][cluster_label] += correct.item()
                num_instance[j][label][cluster_label] += 1

        return num_correct, num_instance

    def n_correct(self, pred, labels):
        _, predicted = torch.max(pred.data, 1)
        n_correct = (predicted == labels).sum().item()
        return n_correct

    def linear_evaluation(self, loader, token='biased_linear', finetune=False):
        scaler = GradScaler(enabled=self.args.fp16_precision)

        logging.info(f"Start Linear evaluation for {self.args.linear_iters} iterations.")
        imagenet_stats = {'biased': [], 'unbiased': [], 'ImageNetA': []}

        for iter_counter in range(self.args.linear_iters):
            inputs = next(loader)
            images, labels = inputs.images, inputs.labels

            with autocast(enabled=self.args.fp16_precision):
                aux = self.nets.encoder(images, freeze=True if not finetune else False, penultimate=True)
                features_penul = aux['penultimate']
                logits = self.nets.classifier(features_penul)
                loss = self.criterion(logits, labels)

            self.optims.classifier.zero_grad()
            if finetune: self.optims.encoder.zero_grad()

            scaler.scale(loss).backward()

            scaler.step(self.optims.classifier)
            if finetune: scaler.step(self.optims.encoder)
            scaler.update()

            if (iter_counter + 1) % self.args.log_every_n_steps == 0:
                top1 = accuracy(logits, labels, topk=(1, ))
                self.writer.add_scalar('loss', loss, global_step=iter_counter+1)
                self.writer.add_scalar('acc/top1', top1[0], global_step=iter_counter+1)
                #self.writer.add_scalar('learning_rate', self.scheduler.classifier.get_lr()[0], global_step=iter_counter+1)

            # warmup for the first 10 epochs
            #if (iter_counter+1) / len(loader) >= self.args.lr_decay_offset:
            #    self.scheduler.classifier.step()

            if (iter_counter+1) % self.args.eval_every == 0:
                if self.args.data != 'imagenet':
                    total_acc, valid_attrwise_acc = self.validation(self.loaders.val)
                    self.report_validation(valid_attrwise_acc, total_acc, iter_counter+1)
                    msg = f"Iter: {iter_counter+1}\tLoss: {loss}\tAccuracy: {total_acc}"
                    logging.info(msg)
                    print(msg)
                else:
                    msg = f"Iter: {iter_counter+1}\t"
                    for key, val_loader in self.loaders.val.items():
                        val_acc = self.validate_imagenet(val_loader, key=key)
                        imagenet_stats[key].append(val_acc)
                        msg += f"{key}: {val_acc}\t"
                    logging.info(msg)
                    print(msg)

        logging.info("Training has finished.")
        # save model checkpoints
        self._save_checkpoint(step=iter_counter+1, token=token)
        torch.save(imagenet_stats, ospj(self.args.log_dir, f'imagenet_valid_acc_{token}.pth'))

        logging.info(f"Model checkpoint and metadata has been saved at {self.args.log_dir}.")

    def measure_rank(self, fetcher):
        self.nets.encoder.eval()
        self.nets.classifier.eval()

        images, _, _, _ = next(iter(fetcher))
        images = images.to(self.device)

        with torch.no_grad():
            aux = self.nets.encoder(images, penultimate=True)
            features = aux['penultimate']
            rank = self.effective_rank(features)
        print(f'rank: {rank}')
        torch.save(rank, ospj(self.args.log_dir, 'rank.pth'))

        self.nets.encoder.train()
        self.nets.classifier.train()

    def train(self):
        try:
            self._load_checkpoint(self.args.simclr_epochs, 'biased_simclr')
            print('Pretrained SimCLR ckpt exists. Move onto linear evaluation')
        except:
            print('Start SimCLR pretraining...')
            self.contrastive_train()
            print('Finished pretraining. Move onto linear evaluation')

        if not self.args.finetune:
            fetcher = InputFetcher(self.loaders.train_linear)
            self.linear_evaluation(fetcher)
        else:
            fetcher = InputFetcher(self.loaders.train_finetune)
            # Set finetune=False. Still freeze encoder during bias-conflict sample identification
            self.linear_evaluation(fetcher, token='biased_finetune', finetune=False if not self.args.finetune_only_once else True)

        if self.args.data != 'imagenet':
            self.save_score_idx(self.loaders.train_linear if not self.args.finetune else self.loaders.train_finetune)
        else:
            self.imagenet_save_score_idx(self.loaders.train_linear)

        fetcher_val = self.loaders.val
        self.measure_rank(fetcher_val)

    def evaluate(self):
        fetcher_val = self.loaders.val
        self._load_checkpoint(self.args.linear_iters, 'biased_linear')
        self.measure_rank(fetcher_val)
        total_acc, valid_attrwise_acc = self.validation(fetcher_val)
        self.report_validation(valid_attrwise_acc, total_acc, 0)
