from subgraphs.densest_subgraph import densest_subgraph
from subgraphs.densest_distinct_subgraphs import densest_distinct_subgraph, find_distinct_densest_subgraphs_iterative
from utils.util import assign_remaining_vertices
import networkx as nx
import numpy as np

def top_k_overlapping_densest_subgraphs(G, k, lambda_param, min_subset_size, max_subset_size, k_hop):
    """
    Identify the top-k overlapping densest subgraphs in a graph.

    This function finds the top-k densest subgraphs in the graph `G` with minimal overlap.
    It uses Charikar's 2-approximation algorithm with diversity constraints to find
    distinct subgraphs efficiently.
    
    Args:
        G (object): A graph represented as networkx graph object
        k (int): The number of densest subgraphs to identify
        lambda_param (float): Number that controls the trade-off between density and diversity of the subgraphs
        min_subset_size (int): Minimum number of vertices in a subgraph
        max_subset_size (int): Maximum number of vertices in a subgraph
        k_hop (int): The maximum distance (in hops) to assign remaining vertices to a subgraph.
    
    Returns:
        List of networkx graph objects
        A list of `k` densest subgraphs, where overlapping vertices are minimized

    """
    
    # Use the new efficient iterative approach
    subgraphs = find_distinct_densest_subgraphs_iterative(
        G, k, lambda_param, min_subset_size, max_subset_size
    )
    
    # If we didn't find enough subgraphs, fall back to the original approach
    # but with the improved densest_subgraph function
    if len(subgraphs) < k:
        # Step 2: Initialize with the densest subgraph
        initial_subgraph = densest_subgraph(G, min_subset_size, max_subset_size)
        W = [initial_subgraph] if initial_subgraph.number_of_nodes() > 0 else []

        while len(W) < k:
            # Step 3: Iteratively compute the next densest distinct subgraph
            next_subgraph = densest_distinct_subgraph(G, W, lambda_param, min_subset_size, max_subset_size)

            if next_subgraph is None or next_subgraph.number_of_nodes() == 0:
                break

            W.append(next_subgraph)
    
    # Assign the remaining vertices to the closest subgraph using k-hop
    W = assign_remaining_vertices(G, subgraphs, k_hop)
    # If we reach k subgraphs or have no more subgraphs to add, return the result
    return W
