from dataclasses import dataclass

from ruamel.yaml import YAML, yaml_object
from typing_extensions import override

from ..._graph_generator_config import GraphGeneratorConfig
from ._generator import generate_simple_graph


@yaml_object(YAML())
@dataclass()
class SimpleGraphConfig(GraphGeneratorConfig):
    """
    Fields:

    - `num_nodes`: Total number of nodes in the graph.
    - `num_clusters`: Number of complete subgraphs to generate.
    - `min_edges_between_clusters`, `max_edges_between_clusters`: The number of edges generated between the complete
                                                                  subgraphs will be drawn from this interval
                                                                  (inclusive).
                                                                  The actual number of edges between clusters might be
                                                                  lower than `min_edges_between_clusters`!
    """

    num_nodes: int
    num_clusters: int
    min_edges_between_clusters: int
    max_edges_between_clusters: int

    # overriding abstract method (no type hint on purpose)
    generate_graph = generate_simple_graph

    @override
    def validate(self):
        min_cluster_size = self.max_edges_between_clusters + 2
        if self.num_clusters * min_cluster_size > self.num_nodes:
            raise ValueError(
                f"num_nodes ({self.num_nodes}) cannot be smaller than "
                "num_clusters * (config.max_edges_between_clusters + 2) "
                f"({self.num_clusters} * {min_cluster_size} = {self.num_clusters * min_cluster_size})"
            )


if __name__ == "__main__":
    from visualise_graph import visualise_graph
    config = SimpleGraphConfig(num_nodes=30, num_clusters=3, min_edges_between_clusters=6, max_edges_between_clusters=8)
    graph = config.generate_graph()
    visualise_graph(graph)
