#!/usr/bin/env python
# -*- coding: utf-8 -*-
# Python version: 3.6


from math import ceil
import numpy as np
from torchvision import datasets, transforms
import torch

from collections import defaultdict


def mnist_iid_normal(dataset, num_users, size):
    """
    Sample I.I.D. client data from MNIST dataset
    :param dataset:
    :param num_users:
    :return: dict of image index
    """
    if size <= 0 or size >= len(dataset):
        num_items = int(len(dataset)/num_users)
    else:
        num_items = int(size/num_users)
    dict_users, all_idxs = {}, [i for i in range(len(dataset))]
    for i in range(num_users):
        train_index = np.random.choice(all_idxs, num_items, replace=False)
        
        all_idxs = list(set(all_idxs) - set(train_index))
        #print(train_index.shape, len(all_idxs))
        #train_index= list(train_index)
        val_index = np.random.choice(train_index, ceil(train_index.shape[0]*0.5), replace=False)
        train_index = np.array(list(set(train_index) - set(val_index)))

        #print(train_index.shape,val_index.shape)
        dict_users[i] = train_index
        dict_users[-i-1] = val_index
        
    return dict_users, dataset


def mnist_iid(dataset, num_users):
    """
    Sample I.I.D. client data from MNIST dataset
    :param dataset:
    :param num_users:
    :return: dict of image index
    """
    trans_mnist = transforms.Compose(
            [transforms.ToTensor(), transforms.Normalize((0.1307,), (0.3081,))])
    dataset_real = datasets.MNIST(
            '../data/mnist/', train=True, download=True, transform=trans_mnist)

    dict_users = {i: np.array([], dtype='int64') for i in range((-num_users-1),num_users)}

    labels = dataset.targets.numpy()

    # Imbalance
    mu=0.1**(1/9)
    probability=[mu**i for i in range(0,10)]
    #print(probability)
    all_idxs = []
    for i in range(10):
        num_sample=min(ceil(labels.shape[0]/10*probability[i]),6000)
        print(num_sample,len(np.where(labels==i)[0][:num_sample]))
        all_idxs.extend(np.where(labels==i)[0][:num_sample])
    print(len(all_idxs))
    for i in range(10):
        labels_new = labels[all_idxs]
        print(np.where(labels_new==i)[0].shape,np.where(labels==i)[0])
    dataset_real.data = dataset_real.data[all_idxs]
    dataset_real.targets = dataset_real.targets[all_idxs]
    # train-val split, for i client : train = dict_user[i], val = dict_user[-i-1]
    num_items = int(len(all_idxs)/num_users)
    dict_users = {}
    for i in range(num_users):
        train_index = np.random.choice(all_idxs, num_items, replace=False)
        
        all_idxs = list(set(all_idxs) - set(train_index))
        #print(train_index.shape, len(all_idxs))
        #train_index= list(train_index)
        val_index = np.random.choice(train_index, ceil(train_index.shape[0]*0.5), replace=False)
        train_index = np.array(list(set(train_index) - set(val_index)))

        #print(train_index.shape,val_index.shape)
        dict_users[i] = train_index
        dict_users[-i-1] = val_index
        
    return dict_users, dataset_real



def mnist_noniid_split(dataset, num_users, size, alpha):

    labels = np.array(dataset.targets) if hasattr(dataset, 'targets') else np.array(dataset.labels)
    num_classes = len(np.unique(labels))
    class_indices = [np.where(labels == y)[0] for y in range(num_classes)]
    
    # 控制总样本数量
    if size <= 0 or size > len(dataset):
        size = len(dataset)

    # 将所有类别样本打散后选出总共 size 个样本参与分配
    all_indices = np.concatenate(class_indices)
    np.random.shuffle(all_indices)
    selected_indices = all_indices[:size]

    # 重建 class_indices 只包含被选中的样本
    selected_labels = labels[selected_indices]
    class_indices = [selected_indices[selected_labels == y] for y in range(num_classes)]

    dict_users = defaultdict(list)

    # Dirichlet 分配
    for c, idxs in enumerate(class_indices):
        if len(idxs) == 0:
            continue
        np.random.shuffle(idxs)
        proportions = np.random.dirichlet(np.repeat(alpha, num_users))
        proportions = (np.cumsum(proportions) * len(idxs)).astype(int)[:-1]
        split_idxs = np.split(idxs, proportions)
        for user, user_idxs in enumerate(split_idxs):
            dict_users[user].extend(user_idxs.tolist())

    # 按每个用户划分 train/val
    final_dict_users = {}
    for user in range(num_users):
        user_indices = np.array(dict_users[user])
        if len(user_indices) == 0:
            final_dict_users[user] = np.array([])
            final_dict_users[-user-1] = np.array([])
            continue
        np.random.shuffle(user_indices)
        val_size = ceil(len(user_indices) * 0.5)
        val_index = user_indices[:val_size]
        train_index = user_indices[val_size:]
        final_dict_users[user] = train_index
        final_dict_users[-user-1] = val_index

    return final_dict_users, dataset


def mnist_noniid_normal(dataset, num_users, size):
    """
    Sample non-I.I.D client data from MNIST dataset
    :param dataset:
    :param num_users:
    :return:
    """

    #dict_users = {i: np.array([], dtype='int64') for i in range((-num_users-1),num_users)}

    labels = dataset.targets.numpy()
    all_idxs = np.arange(len(labels))
    if size<=0 or size>= len(labels):
        num_shards, num_imgs = 2*num_users, int(len(dataset)/(2*num_users)) 
        dataset_idx = all_idxs
    else:
        num_shards, num_imgs = 2*num_users, int(size/(2*num_users))
        dataset_idx = np.random.choice(all_idxs, size, replace=False)
    idx_shard = [i for i in range(num_shards)]
    dict_users = {i: np.array([], dtype='int64') for i in range((-num_users-1),num_users)}

    # sample and sort labels
    idxs_labels = np.vstack((all_idxs, labels))[:,dataset_idx]
    print('idxs_labels is', idxs_labels, 'shape is', idxs_labels.shape)
    idxs_labels = idxs_labels[:,idxs_labels[1,:].argsort()]
    print('idxs_labels is sorted as', idxs_labels)
    idxs = idxs_labels[0,:]

    # print(idxs_labels.shape, idxs_labels)
    
    # divide and assign
    for i in range(num_users):
        rand_set = set(np.random.choice(idx_shard, 2, replace=False))
        idx_shard = list(set(idx_shard) - rand_set)

        train_index = []
        for rand in rand_set:
            train_index.extend(idxs[rand*num_imgs:(rand+1)*num_imgs])

        # print(len(train_index),labels[train_index])
        #train_index= list(train_index)
        train_index=np.array(train_index)
        val_index = np.random.choice(train_index, ceil(train_index.shape[0]*0.1), replace=False)
        train_index = np.array(list(set(train_index) - set(val_index)))

        dict_users[i] = train_index
        dict_users[-i-1] = val_index


    return dict_users, dataset


def mnist_noniid(dataset, num_users, size):
    """
    Sample non-I.I.D client data from MNIST dataset
    :param dataset:
    :param num_users:
    :return:
    """
    trans_mnist = transforms.Compose(
            [transforms.ToTensor(), transforms.Normalize((0.1307,), (0.3081,))])
    dataset_real = datasets.MNIST(
            '../data/mnist/', train=True, download=True, transform=trans_mnist)

    dict_users = {i: np.array([], dtype='int64') for i in range((-num_users-1),num_users)}

    labels = dataset.targets.numpy()

    # Imbalance
    mu=0.1**(1/9)
    probability=[mu**i for i in range(0,10)]
    #print(probability)
    all_idxs = []
    for i in range(10):
        num_sample=min(ceil(labels.shape[0]/10*probability[i]),6000)
        #print(num_sample,len(np.where(labels==i)[0][:num_sample]))
        all_idxs.extend(np.where(labels==i)[0][:num_sample])
    #print(len(all_idxs))
    for i in range(10):
        labels_new = labels[all_idxs]
        # print(np.where(labels_new==i)[0].shape,np.where(labels==i)[0])
    dataset_real.data = dataset_real.data[all_idxs]
    dataset_real.targets = dataset_real.targets[all_idxs]

    # num_shards, num_imgs = 200, int(len(all_idxs)/200)
    if size<=0 or size>= len(labels_new):
        num_shards, num_imgs = 2*num_users, int(len(labels_new)/(2*num_users)) 
        dataset_idx = np.arange(len(labels_new))
    else:
        num_shards, num_imgs = 2*num_users, int(size/(2*num_users))
        dataset_idx = np.random.choice(np.arange(len(labels_new)), size, replace=False)
    idx_shard = [i for i in range(num_shards)]
    dict_users = {i: np.array([], dtype='int64') for i in range((-num_users-1),num_users)}

    # idxs_labels = np.vstack((all_idxs,labels[all_idxs]))
    # idxs_labels = idxs_labels[:,idxs_labels[1,:].argsort()]
    idxs_labels = np.vstack((all_idxs, labels[all_idxs]))[:,dataset_idx]
    print('idxs_labels is', idxs_labels, 'shape is', idxs_labels.shape)
    idxs_labels = idxs_labels[:,idxs_labels[1,:].argsort()]
    print('idxs_labels is sorted as', idxs_labels)
    # print(idxs_labels.shape, idxs_labels)
    idxs = idxs_labels[0,:]
    # divide and assign
    for i in range(num_users):
        rand_set = set(np.random.choice(idx_shard, 2, replace=False))
        idx_shard = list(set(idx_shard) - rand_set)

        train_index = []
        for rand in rand_set:
            train_index.extend(idxs[rand*num_imgs:(rand+1)*num_imgs])

        # print(len(train_index),labels[train_index])
        #train_index= list(train_index)
        train_index=np.array(train_index)
        val_index = np.random.choice(train_index, ceil(train_index.shape[0]*0.2), replace=False)
        train_index = np.array(list(set(train_index) - set(val_index)))

        #print(train_index.shape,val_index.shape)
        dict_users[i] = train_index
        dict_users[-i-1] = val_index

    return dict_users, dataset_real

def cifar_iid(dataset, num_users):
    """
    Sample I.I.D. client data from CIFAR10 dataset
    :param dataset:
    :param num_users:
    :return: dict of image index
    """
    num_items = int(len(dataset)/num_users)
    dict_users, all_idxs = {}, [i for i in range(len(dataset))]
    for i in range(num_users):
        train_index = np.random.choice(all_idxs, num_items, replace=False)
        all_idxs = list(set(all_idxs) - set(train_index))
        val_index = np.random.choice(train_index, ceil(train_index.shape[0] * 0.5), replace=False)
        train_index = np.array(list(set(train_index) - set(val_index)))
        dict_users[i] = train_index
        dict_users[-i - 1] = val_index
        # dict_users[i] = set(np.random.choice(all_idxs, num_items, replace=False))
        # all_idxs = list(set(all_idxs) - dict_users[i])
    return dict_users, dataset

def cifar_noniid_normal(dataset, num_users):
    """
    Sample non-I.I.D client data from MNIST dataset
    :param dataset:
    :param num_users:
    :return:
    """
    labels = np.array(dataset.targets)
    num_shards, num_imgs = 200, 250
    idx_shard = [i for i in range(num_shards)]
    dict_users = {i: np.array([], dtype='int64') for i in range((-num_users - 1), num_users)}
    idxs = np.arange(num_shards * num_imgs)

    # sort labels
    idxs_labels = np.vstack((idxs, labels))
    idxs_labels = idxs_labels[:, idxs_labels[1, :].argsort()]
    idxs = idxs_labels[0, :]

    # divide and assign
    for i in range(num_users):
        rand_set = set(np.random.choice(idx_shard, 2, replace=False))
        idx_shard = list(set(idx_shard) - rand_set)

        train_index = []
        for rand in rand_set:
            train_index.extend(idxs[rand * num_imgs:(rand + 1) * num_imgs])

        train_index = np.array(train_index)
        val_index = np.random.choice(train_index, ceil(train_index.shape[0] * 0.2), replace=False)
        train_index = np.array(list(set(train_index) - set(val_index)))

        dict_users[i] = train_index
        dict_users[-i - 1] = val_index

    return dict_users, dataset

def minmax_dataset(args):
    num_users=args.num_users
    base_s= args.minmax_s
    d= args.d
    n=args.n
    
    s_list=[]
    data=[]
    label = []
    #np.random.seed(base_s)
    dict_users = {i: np.array([], dtype='int64') for i in range((-num_users-1),num_users)}
    if args.iid:
        for _ in range(num_users):
            s=base_s
            s_list.append(s)        
    else:
        for _ in range(num_users):
            s = abs(np.random.uniform(0,2*base_s))
            s_list.append(s)

    for i in range(num_users):
        s = s_list[i]

        # b = np.random.normal(0,s**2,n)
        # b = b - np.mean(b)
        # a = np.random.normal(1,s**2,n)
        # a[a<1]=1
        # A = np.diag(a)

        # b = np.zeros(n)
        # A = np.random.normal(0,s**2,(n,d))

        b = np.random.normal(0,s**2,n)
        a = np.random.uniform(0,1.,n)
        A = np.diag(a)

        data.append(A)
        label.extend(b)
        dict_users[i] = list(range(i*n,i*n+n))
        dict_users[-i-1] = list(range(i*n,i*n+n))

    data= np.vstack(data)
    label = np.vstack(label)
    #print(data.shape,label)
    #print(dict_users)
    #print(s_list)
    tensor_data = torch.Tensor(data)
    tensor_label = torch.Tensor(label)
    dataset = torch.utils.data.TensorDataset(tensor_data,tensor_label)
    #print(dataset)
    return dataset,None,dict_users, None, dataset


def fmnist_iid_normal(dataset, num_users):
    """
    Sample I.I.D. client data from Fashion MNIST dataset
    :param dataset:
    :param num_users:
    :return: dict of image index
    """
    
    num_items = int(len(dataset)/num_users)
    dict_users, all_idxs = {}, [i for i in range(len(dataset))]
    for i in range(num_users):
        train_index = np.random.choice(all_idxs, num_items, replace=False)
        
        all_idxs = list(set(all_idxs) - set(train_index))
        #print(train_index.shape, len(all_idxs))
        #train_index= list(train_index)
        #val_index = np.random.choice(train_index, ceil(train_index.shape[0]*0.5), replace=False)
        #train_index = np.array(list(set(train_index) - set(val_index)))
        val_index=train_index
        #print(train_index.shape,val_index.shape)
        dict_users[i] = train_index
        dict_users[-i-1] = val_index
        
    return dict_users, dataset


def fmnist_noniid_split(dataset, num_users, alpha):
    labels = np.array(dataset.targets) if hasattr(dataset, 'targets') else np.array(dataset.labels)
    num_classes = len(np.unique(labels))

    class_indices = [np.where(labels == y)[0] for y in range(num_classes)]

    dict_users = defaultdict(list)


    for c, idxs in enumerate(class_indices):
        if len(idxs) == 0:
            continue
        np.random.shuffle(idxs)
        proportions = np.random.dirichlet(np.repeat(alpha, num_users))
        proportions = (np.cumsum(proportions) * len(idxs)).astype(int)[:-1]
        split_idxs = np.split(idxs, proportions)
        for user, user_idxs in enumerate(split_idxs):
            dict_users[user].extend(user_idxs.tolist())

    final_dict_users = {}
    for user in range(num_users):
        user_indices = np.array(dict_users[user])
        np.random.shuffle(user_indices)
        val_size = ceil(len(user_indices) * 0.1)
        val_index = user_indices[:val_size]
        train_index = user_indices[val_size:]

        final_dict_users[user] = train_index      
        final_dict_users[-user-1] = val_index      

    return final_dict_users, dataset



def fmnist_noniid_normal(dataset, num_users):
    """
    Sample non-I.I.D client data from MNIST dataset
    :param dataset:
    :param num_users:
    :return:
    """

    #dict_users = {i: np.array([], dtype='int64') for i in range((-num_users-1),num_users)}

    
    labels=dataset.targets.numpy()
    num_shards, num_imgs = num_users*2, 18000//(num_users*2)
    idx_shard = [i for i in range(num_shards)]
    dict_users = {i: np.array([], dtype='int64') for i in range((-num_users-1),num_users)}
    idxs = np.arange(num_shards*num_imgs)

    # sort labels
    idxs_labels = np.vstack((idxs, labels))
    idxs_labels = idxs_labels[:,idxs_labels[1,:].argsort()]
    idxs = idxs_labels[0,:]

    for i in range(num_users):
        rand_set = set(np.random.choice(idx_shard, 2, replace=False))
        idx_shard = list(set(idx_shard) - rand_set)

        train_index = []
        for rand in rand_set:
            train_index.extend(idxs[rand*num_imgs:(rand+1)*num_imgs])

        train_index=np.array(train_index)
        #val_index = np.random.choice(train_index, ceil(train_index.shape[0]*0.2), replace=False)
        #train_index = np.array(list(set(train_index) - set(val_index)))
        val_index=train_index
        dict_users[i] = train_index
        dict_users[-i-1] = val_index

    return dict_users, dataset


if __name__ == '__main__':
    dataset_train = datasets.FashionMNIST("../data/fmnist/",train=True, download=True)
    dataset_test = datasets.FashionMNIST("../data/fmnist", train = False, download=True)
    dataset_train = datasets.MNIST('../data/mnist/', train=True, download=True,
                                   transform=transforms.Compose([
                                       transforms.ToTensor(),
                                       transforms.Normalize((0.1307,), (0.3081,))
                                   ]))
    num = 100
    d = mnist_noniid(dataset_train, num)
