# Importing necessary packages
import networkx as nx


# Defining the density function for our objective function
def density(subgraph):
    """
    The density is the ratio of the total number of edges to the total number of vertices in the subgraph.
    We use .number_of_edges() & .number_of_nodes() methods of networkx package.
    
    Args:
        subgraph (obj): A graph represented as networkx graph object
        
    Returns:
        The density of a give subgraph
        
    """
    if len(subgraph) == 0:
        return float('-inf')
    return subgraph.number_of_edges() / subgraph.number_of_nodes()


# Diameter function
def diameter_within_limit(subgraph, max_diameter):
    """
    The diameter of a graph is the length of the shortest path between the most distanced nodes. 
    This function checks if the diameter of a subgraph is within a specified limit.
    
    Args:
        subgraph (obj): A graph represented as networkx graph object
        max_diameter (int): The maximum allowed diameter 
        
    Returns:
        Boolean value about whether or not the diameter is below max_diameter
    """
    if nx.is_connected(subgraph): # Check if the subgraph is connected
        return nx.diameter(subgraph) <= max_diameter
    return False


# Calculating the distance between two subgraphs
def distance(U, Z):
    """
    Calculate the Jaccard distance between two subgraphs based on their node sets.
    This is a proper distance metric that satisfies:
    - Symmetry: d(U,Z) = d(Z,U)
    - Triangle inequality: d(U,V) + d(V,Z) >= d(U,Z)
    - Non-negativity: d(U,Z) >= 0
    - Identity: d(U,U) = 0
    
    Args:
        U & Z: Two subgraphs as networkx graph object
        
    Returns:
        The Jaccard distance between the node sets of the two subgraphs (0 to 1)
    """
    if len(U.nodes) == 0 or len(Z.nodes) == 0:
        return 1.0  # Maximum distance for empty subgraphs

    U_set = set(U.nodes)
    Z_set = set(Z.nodes)

    if U_set == Z_set:
        return 0.0  # Identical subgraphs

    # Jaccard distance = 1 - Jaccard similarity
    intersection_size = len(U_set & Z_set)
    union_size = len(U_set | Z_set)
    
    if union_size == 0:
        return 1.0
    
    return 1.0 - (intersection_size / union_size)


# Identifying the radius
def k_hop_neighbors(G, node, k):
    """ 
    The k-hop neighbors of a given node.
    Acts as a helper function to assign_remaining_vertices function.
    
    Args: 
        G (object): A graph represented as networkx graph object
        node: Starting node for path
        k: depth to stop the search
        
    Returns:
        All the nodes within k-hops (shortest path distance) from a give node
    """
    return set(nx.single_source_shortest_path_length(G, node, cutoff=k).keys())


def assign_remaining_vertices(G, subgraphs, k):
    """
    Assigns remaining nodes to the nearest subgraph based on k-hop distance.
    
    Args:
        G (nx.Graph): The original graph represented as a NetworkX graph object.
        subgraphs (list): A list of subgraphs represented as NetworkX graph objects.
        k (int): Maximum distance (in hops) to consider for assigning nodes.
        
    Returns:
        list: Updated list of subgraphs with added nodes.
    """
    remaining_nodes = set(G.nodes()) - set(node for sg in subgraphs for node in sg.nodes())
    for node in remaining_nodes:
        # Initialize the closest subgraph and distance
        min_distance = float('inf')
        closest_subgraph = None
        
        # Iterate through subgraphs to find the nearest one
        for subgraph in subgraphs:
            subgraph_nodes = subgraph.nodes()
            try:
                # Compute distances from the current node to all subgraph nodes
                distances = [
                    nx.shortest_path_length(G, source=node, target=sg_node)
                    for sg_node in subgraph_nodes
                ]
                # Find the minimum distance for the current subgraph
                min_subgraph_distance = min(distances)
                if min_subgraph_distance < min_distance:
                    min_distance = min_subgraph_distance
                    closest_subgraph = subgraph
            except nx.NetworkXNoPath:
                # Ignore if no path exists to any node in this subgraph
                continue

        # Add the node to the closest subgraph if within k hops
        if closest_subgraph and min_distance <= k:
            closest_subgraph.add_node(node)
            for neighbor in G.neighbors(node):
                if neighbor in closest_subgraph.nodes():
                    closest_subgraph.add_edge(node, neighbor)

    return subgraphs
