import os
os.environ["CUDA_VISIBLE_DEVICES"] = "0"
import argparse

import torch
import torch.nn as nn
import torch.optim as optim
import torchvision
import torchvision.transforms as transforms

import matplotlib.pyplot as plt
import numpy as np
import pickle

from utils import *
from models import *

class MyDataset(torch.utils.data.Dataset):
	def __init__(self, data, target, transform=None):
		self.data = data
		self.target = target
		self.transform = transform
		
	def __len__(self):
		return len(self.data)
	
	def __getitem__(self, index):
		x = self.data[index]
		if self.transform:
			x = self.transform(x)
		y = self.target[index]
		return x, y

def sampling(model, device, pool_loader, select_samples, list_bins, n_bins = 1000):
	model.eval()
	outputs, targets = [], []
	with torch.no_grad():
		for data, target in pool_loader:
			data, target = data.to(device), target.to(device)
			output = model(data)
			output = torch.softmax(output, dim=1)
			outputs.append(output.cpu().numpy())
			targets.append(target.cpu().numpy())
	
	tmps = np.asarray(outputs[:len(outputs)-1])
	tmps = tmps.reshape(tmps.shape[0] * tmps.shape[1], tmps.shape[2])
	outputs = np.concatenate((tmps, outputs[len(outputs)-1]), 0)

	out = []
	outputs_max = outputs.max(1)
	for binn in list_bins:
		lower_bin, higher_bin = binn/n_bins, (binn+1)/n_bins
		for idx in range(len(outputs_max)):
			if outputs_max[idx] >= lower_bin and outputs_max[idx] <= higher_bin:
				out.append(idx)
				if len(out) == select_samples:
					return np.array(out)

def estimate_ece(softmaxes, labels, n_bins=1000):
	softmaxes = torch.tensor(softmaxes)
	labels = torch.tensor(labels)
	bin_boundaries = torch.linspace(0, 1, n_bins + 1)
	bin_lowers = bin_boundaries[:-1]
	bin_uppers = bin_boundaries[1:]

	confidences, predictions = torch.max(softmaxes, 1)
	accuracies = predictions.eq(labels)

	list_ece = []
	t = 0
	for bin_lower, bin_upper in zip(bin_lowers, bin_uppers):
		in_bin = confidences.gt(bin_lower.item()) * confidences.le(bin_upper.item())
		prop_in_bin = in_bin.float().mean()
		accuracy_in_bin = accuracies[in_bin].float().mean()
		avg_confidence_in_bin = confidences[in_bin].mean()
		list_ece.append(torch.abs(avg_confidence_in_bin - accuracy_in_bin) * prop_in_bin)
	
	list_ece = np.array(list_ece)
	out = list_ece.argsort()
	return out[::-1]

def sampling_balance(pool_labels):
	out = []
	while True:
		idx = np.random.choice(pool_labels.shape[0], 1)[0]
		if idx not in out:
			count = 0
			for i_out in out:
				if pool_labels[i_out] == pool_labels[idx]:
					count += 1
			
			if count < 10:
				out.append(idx)
		
		if len(out) == 100:
			return np.array(out)

if __name__=="__main__":
	parser = argparse.ArgumentParser()
	parser.add_argument("--exp_idx", help="Index of experiment")
	args = parser.parse_args()
	fix_random_seed(int(args.exp_idx))
	device = 'cuda' 

	train_epochs = 30
	select_samples = 100
	train_data_path = 'data/cifar10_train_data.npy'
	train_labels_path = 'data/cifar10_train_labels.npy'
	test_data_path = 'data/cifar10_test_data.npy'
	test_labels_path = 'data/cifar10_test_labels.npy'
	test_c_data_path = 'data/CIFAR-10-C/gaussian_noise.npy'
	test_c_labels_path = 'data/CIFAR-10-C/labels.npy'

	net = ResNet18()
	net = net.to(device)

	transform = transforms.Compose([
		transforms.ToTensor(),
		transforms.Normalize((0.4914, 0.4822, 0.4465), (0.2023, 0.1994, 0.2010)),
	])

	data = np.load(test_data_path) 
	labels = np.load(test_labels_path)
	testloader = torch.utils.data.DataLoader(MyDataset(data, labels, transform=transform), batch_size=100, shuffle=False)

	data_c = np.load(test_c_data_path)
	data_c = (data_c / 255).astype(np.float32)
	labels_c = np.load(test_c_labels_path)
	testloader_c = torch.utils.data.DataLoader(MyDataset(data_c, labels_c, transform=transform), batch_size=100, shuffle=False)

	pool_data = np.load(train_data_path)
	pool_labels = np.load(train_labels_path)
	pool_loader = torch.utils.data.DataLoader(MyDataset(pool_data, pool_labels, transform=transform), batch_size=128, shuffle=False)

	list_selected_data, list_selected_labels = [], []
	list_acc, list_ece, list_acc_c, list_ece_c = [], [], [], []
	for idx in range(9):
		idxs_unlabeled = sampling_balance(pool_labels)
		list_selected_data.append(pool_data[idxs_unlabeled])
		list_selected_labels.append(pool_labels[idxs_unlabeled])
		pool_data = np.delete(pool_data, idxs_unlabeled, 0)
		pool_labels = np.delete(pool_labels, idxs_unlabeled)
	idxs_unlabeled = sampling_balance(pool_labels)
	
	for rd in range(100):
		list_selected_data.append(pool_data[idxs_unlabeled])
		list_selected_labels.append(pool_labels[idxs_unlabeled])
		selected_data = np.asarray(list_selected_data)
		selected_data = np.reshape(selected_data, (selected_data.shape[0]*select_samples, 32, 32, 3))
		selected_labels = np.asarray(list_selected_labels)
		selected_labels = np.reshape(selected_labels, (selected_labels.shape[0]*select_samples,))
		trainloader = torch.utils.data.DataLoader(MyDataset(selected_data, selected_labels, transform=transform), batch_size=128, shuffle=True)
		
		criterion = nn.CrossEntropyLoss()
		optimizer = optim.Adam(net.parameters())
		for epoch in range(train_epochs):
			train(net, device, trainloader, criterion, optimizer)

		outputs, targets, acc = test_model(net, device, testloader)
		list_bins = estimate_ece(outputs, targets)

		pool_data = np.delete(pool_data, idxs_unlabeled, 0)
		pool_labels = np.delete(pool_labels, idxs_unlabeled)
		pool_loader = torch.utils.data.DataLoader(MyDataset(pool_data, pool_labels, transform=transform), batch_size=512, shuffle=False)

		idxs_unlabeled = sampling(net, device, pool_loader, select_samples, list_bins)

		outputs, targets, acc = test_model(net, device, testloader)
		ece = calculate_ece(torch.tensor(outputs), torch.tensor(targets))
		list_acc.append(acc)
		list_ece.append(ece)

		outputs, targets, acc = test_model(net, device, testloader_c)
		ece = calculate_ece(torch.tensor(outputs), torch.tensor(targets))
		list_acc_c.append(acc)
		list_ece_c.append(ece)

		with open("out/cifar10/demo_ECE" + str(args.exp_idx), "wb") as fp:
			pickle.dump(list_acc, fp)
			pickle.dump(list_ece, fp)
			pickle.dump(list_acc_c, fp)
			pickle.dump(list_ece_c, fp)