from config import cfg
import torch
import torchvision
import torch.nn as nn
from torch.autograd import Variable
from torch.nn.parameter import Parameter
import torch.nn.functional as F
from torchvision import transforms
import numpy as np
import pandas as pd
import os
import argparse
from tqdm import tqdm, trange
import torch.nn.functional as F
from resnetCopy1 import ResNet18
import glob
import torchvision.models as models
import trojanvision
from resnet import ResNet 
from mobilenet import mobilenetv2
from wresnet import WideResNet
from mobilevit import MobileViTv2
from swin import SwinTransformer
import torchvision.datasets as datasets
import torch.utils.data as data

class Program(nn.Module):
    def __init__(self, cfg, gpu):
        super(Program, self).__init__()
        self.cfg = cfg
        self.gpu = gpu
        self.init_mask()
        self.W = Parameter(torch.randn(self.M.shape), requires_grad=True)
        
    def init_net(self, i):
        if self.cfg.net == 'clean':
            # self.net = ResNet18(num_classes=100)
            self.net = mobilenetv2(num_classes=100)
            mean = np.array([0.485, 0.456, 0.406], dtype=np.float32)
            mean = mean[..., np.newaxis, np.newaxis]
            std = np.array([0.229, 0.224, 0.225],dtype=np.float32)
            std = std[..., np.newaxis, np.newaxis]
            self.filename = '{:04d}'.format(i)
            print(self.filename)
            print(self.cfg.net)
            checkpoint = torch.load(os.path.join(self.cfg.models_dir, self.cfg.exp_name, 'clean_' + self.filename + '.pt'), map_location='cuda:0')
            #checkpoint = torch.load(os.path.join(self.cfg.models_dir, 'clean_gt', 'cleangt_' + self.filename + '.pt'), map_location='cuda:0')
            self.net.load_state_dict(checkpoint)
            device = torch.device("cuda:0" if torch.cuda.is_available() else "cpu")
            self.mean = Parameter(torch.from_numpy(mean).to(device), requires_grad=False)
            self.std = Parameter(torch.from_numpy(std).to(device), requires_grad=False)

            
        elif self.cfg.net == 'backdoor':
            # self.net = ResNet18(num_classes=100)
            self.net = mobilenetv2(num_classes=100)
            mean = np.array([0.485, 0.456, 0.406], dtype=np.float32)
            mean = mean[..., np.newaxis, np.newaxis]
            std = np.array([0.229, 0.224, 0.225],dtype=np.float32)
            std = std[..., np.newaxis, np.newaxis]
            self.filename = '{:04d}'.format(i)
            print(self.filename)
            print(self.cfg.net)
            checkpoint = torch.load(os.path.join(self.cfg.models_dir, self.cfg.exp_name, 'backdoor_' + self.filename + '.pt'), map_location='cuda:0')
            #checkpoint = torch.load(os.path.join(self.cfg.models_dir, 'badnet_gt_0', 'badnet_gt_' + self.filename + '.pt'), map_location='cuda:0')
            self.net.load_state_dict(checkpoint)
            device = torch.device("cuda:0" if torch.cuda.is_available() else "cpu")
            self.mean = Parameter(torch.from_numpy(mean).to(device), requires_grad=False)
            self.std = Parameter(torch.from_numpy(std).to(device), requires_grad=False)

            
        else:
            raise NotImplementedError()

        self.net.to(torch.device('cuda'))
        for param in self.net.parameters():
            param.requires_grad = False


    def init_mask(self):
        M = torch.ones(3, self.cfg.h1, self.cfg.w1)
        c_w, c_h = int(np.ceil(self.cfg.w1/2.)), int(np.ceil(self.cfg.h1/2.))
        M[:,c_h-self.cfg.h2//2:c_h+self.cfg.h2//2, c_w-self.cfg.w2//2:c_w+self.cfg.w2//2] = 0
        self.M = Parameter(M, requires_grad=False)

    def imagenet_label2_mnist_label(self, imagenet_label):
        # return imagenet_label[:,:10]
        return imagenet_label[:,:100]

    def forward(self, image):
        if self.cfg.dataset == 'mnist':
            image = image.repeat(1,3,1,1)
            
        elif self.cfg.dataset == 'cifar10' or self.cfg.dataset == 'cifar100':
            image = image.repeat(1,1,1,1)
            
        elif self.cfg.dataset == 'STL10' or self.cfg.dataset == 'tinyimagenet':
            image = image.repeat(1,1,1,1)
            
        
        X = image.data.new(self.cfg.batch_size_per_gpu, 3, self.cfg.h1, self.cfg.w1)
        X[:] = 0
        X[:,:,int((self.cfg.h1-self.cfg.h2)//2):int((self.cfg.h1+self.cfg.h2)//2), int((self.cfg.w1-self.cfg.w2)//2):int((self.cfg.w1+self.cfg.w2)//2)] = image.data.clone()
        X = Variable(X, requires_grad=True)
        P = torch.sigmoid(self.W * self.M)
        X_adv = X + P
        X_adv = (X_adv - self.mean) / self.std
        Y_adv = self.net(X_adv)
        Y_adv = F.softmax(Y_adv, 1)
        return self.imagenet_label2_mnist_label(Y_adv)


class Adversarial_Reprogramming(object):
    def __init__(self, args, cfg=cfg):
        self.mode = args.mode
        self.gpu = args.gpu
        self.restore = args.restore
        self.shadow = args.shadow
        self.cfg = cfg
        self.cfg.exp_name = args.exp_name
        self.init_dataset()
        self.Program = Program(self.cfg, self.gpu)
        self.restore_from_file()
        self.set_mode_and_gpu()

    def init_dataset(self):
        seed = 42
        torch.manual_seed(seed)
        rng = torch.Generator()
        rng.manual_seed(seed)
        data_transform = transforms.Compose([
                transforms.Resize([20,20]),
                transforms.ToTensor()
            
            ])
        
        
        if self.cfg.dataset == 'mnist':
            train_set = torchvision.datasets.MNIST(os.path.join(self.cfg.data_dir, 'mnist'), train=True, transform=transforms.ToTensor(), download=True)
            test_set = torchvision.datasets.MNIST(os.path.join(self.cfg.data_dir, 'mnist'), train=False, transform=transforms.ToTensor(), download=True)
            #test_sampler = torch.utils.data.RandomSampler(test_set, num_samples=len(test_set), replacement=False, generator=torch.Generator().manual_seed(42))
            kwargs = {'num_workers': 1, 'pin_memory': True, 'drop_last': True}
            #indices = [i for i, (x, y) in enumerate(test_set) if y == 0]
            #test_set = torch.utils.data.Subset(test_set, indices)
            if self.gpu:
                self.train_loader = torch.utils.data.DataLoader(train_set, batch_size=self.cfg.batch_size_per_gpu*len(self.gpu), shuffle=True,  **kwargs)
                self.test_loader = torch.utils.data.DataLoader(test_set, generator=rng, batch_size=self.cfg.batch_size_per_gpu*len(self.gpu),  **kwargs)
            else:
                self.train_loader = torch.utils.data.DataLoader(train_set, batch_size=self.cfg.batch_size_per_gpu, shuffle=True,  **kwargs)
                self.test_loader = torch.utils.data.DataLoader(test_set, generator=rng, batch_size=self.cfg.batch_size_per_gpu,  **kwargs)
                
        elif self.cfg.dataset == 'cifar10':
            train_set = torchvision.datasets.CIFAR10(os.path.join(self.cfg.data_dir, 'cifar10'), train=True, transform=transforms.ToTensor(), download=True)
            test_set = torchvision.datasets.CIFAR10(os.path.join(self.cfg.data_dir, 'cifar10'), train=False, transform=transforms.ToTensor(), download=True)
            #test_sampler = torch.utils.data.RandomSampler(test_set, num_samples=len(test_set), replacement=False, generator=torch.Generator().manual_seed(42))
            kwargs = {'num_workers': 1, 'pin_memory': True, 'drop_last': True}
            #test_indices = np.where(np.array(test_set.targets) == 0)[0]
            #test_set = torch.utils.data.Subset(test_set, test_indices)
            if self.gpu:
                self.train_loader = torch.utils.data.DataLoader(train_set, batch_size=self.cfg.batch_size_per_gpu*len(self.gpu), shuffle=True,  **kwargs)
                self.test_loader = torch.utils.data.DataLoader(test_set, generator=rng, batch_size=self.cfg.batch_size_per_gpu*len(self.gpu),  **kwargs)
            else:
                self.train_loader = torch.utils.data.DataLoader(train_set, batch_size=self.cfg.batch_size_per_gpu, shuffle=True,  **kwargs)
                self.test_loader = torch.utils.data.DataLoader(test_set, generator=rng, batch_size=self.cfg.batch_size_per_gpu,  **kwargs)
        
        elif self.cfg.dataset == 'cifar100':
            train_set = torchvision.datasets.CIFAR100(os.path.join(self.cfg.data_dir, 'cifar100'), train=True, transform=transforms.ToTensor(), download=True)
            test_set = torchvision.datasets.CIFAR100(os.path.join(self.cfg.data_dir, 'cifar100'), train=False, transform=transforms.ToTensor(), download=True)
            #test_sampler = torch.utils.data.RandomSampler(test_set, num_samples=len(test_set), replacement=False, generator=torch.Generator().manual_seed(42))
            kwargs = {'num_workers': 1, 'pin_memory': True, 'drop_last': True}
            #test_indices = np.where(np.array(test_set.targets) == 0)[0]
            #test_set = torch.utils.data.Subset(test_set, test_indices)
            if self.gpu:
                self.train_loader = torch.utils.data.DataLoader(train_set, batch_size=self.cfg.batch_size_per_gpu*len(self.gpu), shuffle=True,  **kwargs)
                self.test_loader = torch.utils.data.DataLoader(test_set, generator=rng, batch_size=self.cfg.batch_size_per_gpu*len(self.gpu),  **kwargs)
            else:
                self.train_loader = torch.utils.data.DataLoader(train_set, batch_size=self.cfg.batch_size_per_gpu, shuffle=True,  **kwargs)
                self.test_loader = torch.utils.data.DataLoader(test_set, generator=rng, batch_size=self.cfg.batch_size_per_gpu,  **kwargs)  
                  
        elif self.cfg.dataset == 'STL10':
            train_set = torchvision.datasets.STL10(os.path.join(self.cfg.data_dir, 'STL10'), split='train', transform=transforms.ToTensor(), download=True)
            test_set = torchvision.datasets.STL10(os.path.join(self.cfg.data_dir, 'STL10'), split='test', transform=transforms.ToTensor() , download=True)
            #test_sampler = torch.utils.data.RandomSampler(test_set, num_samples=len(test_set), replacement=False, generator=torch.Generator().manual_seed(42))
            kwargs = {'num_workers': 1, 'pin_memory': True, 'drop_last': True}
            #test_indices = np.where(np.array(test_set.targets) == 0)[0]
            #test_set = torch.utils.data.Subset(test_set, test_indices)
            if self.gpu:
                self.train_loader = torch.utils.data.DataLoader(train_set, batch_size=self.cfg.batch_size_per_gpu*len(self.gpu), shuffle=True,  **kwargs)
                self.test_loader = torch.utils.data.DataLoader(test_set, generator=rng, batch_size=self.cfg.batch_size_per_gpu*len(self.gpu),  **kwargs)
            else:
                self.train_loader = torch.utils.data.DataLoader(train_set, batch_size=self.cfg.batch_size_per_gpu, shuffle=True,  **kwargs)
                self.test_loader = torch.utils.data.DataLoader(test_set, generator=rng, batch_size=self.cfg.batch_size_per_gpu,  **kwargs)
                
        elif self.cfg.dataset == 'tinyimagenet':
            # data_dir = "/home/iis519409/github/pytorch-tiny-imagenet/tiny-imagenet-200/"
            data_dir = os.path.join(self.cfg.data_dir, 'tiny-imagenet-100')
            num_workers = {"train": 2, "val": 0, "test": 0}
            data_transforms = {
                "train": transforms.Compose(
                    [
                        transforms.Resize((self.cfg.h2, self.cfg.w2)),
                        transforms.ToTensor(),
                    ]
                ),
                "val": transforms.Compose(
                    [
                        transforms.Resize((self.cfg.h2, self.cfg.w2)),
                        transforms.ToTensor(),
                    ]
                ),
            }
            
            image_datasets = {x: datasets.ImageFolder(os.path.join(data_dir, x), data_transforms[x]) for x in ["train", "val"]}
            kwargs = {'num_workers': 1, 'pin_memory': True, 'drop_last': True}
            
            if self.gpu:
                self.train_loader = data.DataLoader(image_datasets["train"], batch_size=self.cfg.batch_size_per_gpu*len(self.gpu), shuffle=True,  **kwargs)
                self.test_loader = data.DataLoader(image_datasets["val"], generator=rng, batch_size=self.cfg.batch_size_per_gpu*len(self.gpu),  **kwargs)
            else:
                self.train_loader = data.DataLoader(image_datasets["train"], batch_size=self.cfg.batch_size_per_gpu*len(self.gpu), shuffle=True,  **kwargs)
                self.test_loader = data.DataLoader(image_datasets["val"], generator=rng, batch_size=self.cfg.batch_size_per_gpu*len(self.gpu),  **kwargs)
            
        else:
            raise NotImplementedError()

    def restore_from_file(self):
        if self.restore is not None:
            ckpt = os.path.join(self.cfg.train_dir, 'W_%03d.pt' % self.restore)
            assert os.path.exists(ckpt)
            if self.gpu:
                self.Program.load_state_dict(torch.load(ckpt), strict=False)
            else:
                self.Program.load_state_dict(torch.load(ckpt, map_location='cpu'), strict=False)
            self.start_epoch = self.restore + 1
        else:
            self.start_epoch = 1

    def set_mode_and_gpu(self):
        if self.mode == 'train':
            # optimizer
            self.BCE = torch.nn.BCELoss()
            self.optimizer = torch.optim.Adam(filter(lambda p: p.requires_grad, self.Program.parameters()), lr=self.cfg.lr, betas=(0.9, 0.999))
            self.lr_scheduler = torch.optim.lr_scheduler.StepLR(self.optimizer, step_size=2, gamma=self.cfg.decay)

            if self.gpu:
                with torch.cuda.device(0):
                    self.BCE.cuda()
                    self.Program.cuda()

            if len(self.gpu) > 1:
                self.Program = torch.nn.DataParallel(self.Program, device_ids=list(range(len(self.gpu))))

        elif self.mode == 'validate' or self.mode == 'test':
            if self.gpu:
                with torch.cuda.device(0):
                    self.Program.cuda()

            if len(self.gpu) > 1:
                self.Program = torch.nn.DataParallel(self.Program, device_ids=list(range(len(self.gpu))))
        else:
            raise NotImplementedError()

    @property
    def get_W(self):
        for p in self.Program.parameters():
            if p.requires_grad:
                return p

    def imagenet_label2_mnist_label(self, imagenet_label):
        # return imagenet_label[:,:10]
        return imagenet_label[:,:100]

    def tensor2var(self, tensor, requires_grad=False, volatile=False):
        if self.gpu:
            with torch.cuda.device(0):
                tensor = tensor.cuda()
        return Variable(tensor, requires_grad=requires_grad, volatile=volatile)

    def compute_loss(self, out, label):
        if self.gpu:
            # label = torch.zeros(self.cfg.batch_size_per_gpu*len(self.gpu), 10).scatter_(1, label.view(-1,1), 1)
            label = torch.zeros(self.cfg.batch_size_per_gpu*len(self.gpu), 100).scatter_(1, label.view(-1,1), 1)
        else:
            # label = torch.zeros(self.cfg.batch_size_per_gpu, 10).scatter_(1, label.view(-1,1), 1)
            label = torch.zeros(self.cfg.batch_size_per_gpu, 100).scatter_(1, label.view(-1,1), 1)
        label = self.tensor2var(label)
        return self.BCE(out, label) + self.cfg.lmd * torch.norm(self.get_W) ** 2
    

    def validate(self):
        acc = 0.0
        processed_images = 0
        for k, (image, label) in enumerate(self.test_loader):
            image = self.tensor2var(image)
            out = self.Program(image)
            pred = out.data.cpu().numpy().argmax(1)
            acc += sum(label.numpy() == pred) / float(len(label) * len(self.test_loader))
            processed_images += len(label)
            #if processed_images > (len(self.test_loader.dataset) - 3500):
            out_txt = pd.DataFrame(out.data.cpu().numpy())
            label_txt = pd.DataFrame(label.data.cpu().numpy())
            df = out_txt.assign(label = label_txt)
            if self.cfg.dataset == 'mnist' or 'STL10' or 'tinyimagenet':
                if self.cfg.net == 'clean':
                    df.to_csv(os.path.join(self.cfg.models_dir, self.cfg.exp_name, 'c_train.csv'), mode='a', header=False, index=False)
                elif self.cfg.net == 'backdoor':
                    df.to_csv(os.path.join(self.cfg.models_dir, self.cfg.exp_name, 'b_train.csv'), mode='a', header=False, index=False)

            elif self.cfg.dataset == 'cifar10':
                if self.cfg.net == 'clean':
                    df.to_csv(os.path.join(self.cfg.models_dir, self.cfg.exp_name, 'c_train.csv'), mode='a', header=False, index=False)
                elif self.cfg.net == 'backdoor':
                    df.to_csv(os.path.join(self.cfg.models_dir, self.cfg.exp_name, 'b_train.csv'), mode='a', header=False, index=False)

        print('test accuracy: %.6f' % acc)
        
    def train(self):
        for i in range(self.shadow):
            self.Program.init_net(i)
            for self.epoch in range(self.start_epoch, self.cfg.max_epoch + 1):
                self.lr_scheduler.step()
                for j, (image, label) in enumerate(self.train_loader):
                    if j > 3: break;
                    image = self.tensor2var(image)
                    self.out = self.Program(image)
                    self.loss = self.compute_loss(self.out, label)
                    self.optimizer.zero_grad()
                    self.loss.backward()
                    self.optimizer.step()    
    
                print('net: %d, epoch: %03d/%03d, loss: %.6f' % (i, self.epoch, self.cfg.max_epoch, self.loss.data.cpu().numpy()))
                #torch.save({'W': self.get_W}, os.path.join(self.cfg.train_dir, 'W_net%d_%03d.pt' % (i, self.epoch)))

                if self.epoch == self.cfg.max_epoch:
                    self.validate()
            #self.lr_scheduler.step()
            self.optimizer.param_groups[0]['lr'] = self.cfg.lr
            
    def test(self):
        pass

def main():
    parser = argparse.ArgumentParser()
    parser.add_argument('-m', '--mode', default='train', type=str, choices=['train', 'validate', 'test'])
    parser.add_argument('-r', '--restore', default=None, action='store', type=int, help='Specify checkpoint id to restore.')
    parser.add_argument('-g', '--gpu', default=[], nargs='+', type=str, help='Specify GPU ids.')
    parser.add_argument('-n', '--net', default='clean', type=str, choices=['clean', 'backdoor'])
    parser.add_argument('-s', '--shadow', default=15, type=int)
    parser.add_argument('-e', '--exp_name', type=str, required=True)
    # test params

    args = parser.parse_args()
    cfg['net'] = args.net
    # print(args)
    os.environ['CUDA_VISIBLE_DEVICES'] = ','.join(args.gpu)
    AR = Adversarial_Reprogramming(args)
    if args.mode == 'train':
        AR.train()
    elif args.mode == 'validate':
        AR.validate()
    elif args.mode == 'test':
        AR.test()
    else:
        raise NotImplementedError()

if __name__ == "__main__":
    main()
