import hashlib
import numpy as np
import random
from .topology import UndirectedTopology
from .ring import Ring

def efficient_hash(clientID, coordination):
    """
    Create a hash of the clientID and coordination
    Parameters:
        clientID: Integer (IP address or ID)
        coordination: Integer

    return: String
    """
    
    combined = f"{coordination}, {clientID}"
    hash = hashlib.md5(combined.encode()).hexdigest()
    value = int(hash, 16) / (2**128-1)
    # print(coordination, clientID, value)
    return value


def group_hash(n_clients, n_groups):
    """
    Create a hash of the clientID and coordination
    Parameters:
        n_clients: Integer (Number of clients)
        n_groups: Integer (Number of groups)

    return: List of Integer
    """
    group = [[0] * n_clients for _ in range(n_groups)]
    for i in range(n_groups):
        for j in range(n_clients):
            group[i][j] = efficient_hash(i, j)
    # print(group)
    return group

class FedLay(object):
    def __init__(self, n_clients:int = None, n_groups:int = None, sample_type:str = 'ring'):
        self.n_clients = n_clients
        self.n_groups = n_groups
        self.rings = [None] * n_groups
        self.neighbours = [set() for i in range(self.n_clients)]
        self.overlay = None
        self.__create_topology()
        self.sample_type = sample_type

        

    def __create_topology(self):
        group = group_hash(self.n_clients, self.n_groups)
        for i in range(self.n_groups):
            self.rings[i] = Ring(self.n_clients, group[i])
            for j in range(self.n_clients):
                self.neighbours[j].update(self.rings[i].get_neighbours(j))
        self.overlay = OverLay(self.n_clients, self.neighbours)
        

    def mixing_matrix(self, n_neighbours = 2):
        if self.sample_type == 'ring':
            group = np.random.randint(0, self.n_groups)
            return self.rings[group].mixing_matrix()
        else:
            return self.overlay.mixing_matrix(n_neighbours)
        

class OverLay(UndirectedTopology):
    def __init__(self, n_clients:int = None, neighbours = None):
        super().__init__(n_clients)
        self.dynamic = True
        self.neighbours = neighbours
    def __create_topology(self, n_neighbours):
        for i in range(self.n_clients):
            self.adj_list[i].clear()
            sampled_nodes = random.sample(list(self.neighbours[i]), n_neighbours)
            for n in sampled_nodes:
                self.__insert_edge__(i, n)

    def mixing_matrix(self, n_neighbours):
        self.__create_topology(n_neighbours)
        return super().mixing_matrix()
    
if __name__ == "__main__":
    n_clients = 500
    n_groups = 3
    topology = FedLay(n_clients, n_groups, 'ring')
    for _ in range(3):
        neighbor_counts = (topology.mixing_matrix() > 0).sum(dim=1).tolist()
        print(neighbor_counts)