"""
Complete DAG generator.

A complete DAG has an edge between every pair of nodes,
with direction determined by a topological ordering.
"""

from typing import Optional, List
import numpy as np

from ..core.dag import DAG


def generate_complete_dag(
    d: int,
    ordering: Optional[List[int]] = None
) -> DAG:
    """
    Generate a complete DAG on d nodes.

    For a complete DAG, every pair of nodes has a directed edge.
    The direction is determined by the topological ordering:
    edge i -> j if ordering.index(i) < ordering.index(j)

    Args:
        d: Number of nodes
        ordering: Topological ordering (default: [0, 1, ..., d-1])

    Returns:
        Complete DAG with d(d-1)/2 edges

    Properties:
        - d(d-1)/2 edges
        - Many v-structures: for each pair (i,j) with i<j<k in ordering,
          we have v-structure i -> k <- j
        - MEC size = 1 (all edges compelled by v-structures)
    """
    if d < 1:
        raise ValueError(f"Number of nodes must be positive, got {d}")

    if ordering is None:
        ordering = list(range(d))
    else:
        if len(ordering) != d:
            raise ValueError(f"Ordering must have {d} elements, got {len(ordering)}")
        if set(ordering) != set(range(d)):
            raise ValueError("Ordering must be a permutation of [0, 1, ..., d-1]")

    dag = DAG(d)

    # Create position map: position[node] = index in ordering
    position = {node: idx for idx, node in enumerate(ordering)}

    # Add edge from earlier to later in ordering
    for i in range(d):
        for j in range(d):
            if i != j:
                if position[i] < position[j]:
                    dag.add_edge(i, j)

    return dag


def generate_random_complete_dag(d: int, random_state: Optional[int] = None) -> DAG:
    """
    Generate a complete DAG with random topological ordering.

    Args:
        d: Number of nodes
        random_state: Random seed

    Returns:
        Complete DAG with random ordering
    """
    rng = np.random.default_rng(random_state)
    ordering = list(rng.permutation(d))
    return generate_complete_dag(d, ordering)


def count_complete_dag_edges(d: int) -> int:
    """Return the number of edges in a complete DAG on d nodes."""
    return d * (d - 1) // 2


def count_complete_dag_v_structures(d: int) -> int:
    """
    Count v-structures in a complete DAG.

    For a complete DAG with ordering, every triple (i, j, k) where
    i and j are both parents of k forms a v-structure if i and j
    are both earlier than k in the ordering.

    Number of v-structures = (d choose 3) = d(d-1)(d-2)/6
    """
    if d < 3:
        return 0
    return d * (d - 1) * (d - 2) // 6


def is_complete_dag(dag: DAG) -> bool:
    """
    Check if a DAG is complete.

    A complete DAG has exactly d(d-1)/2 edges.
    """
    d = dag.num_nodes()
    expected_edges = d * (d - 1) // 2
    return dag.num_edges() == expected_edges


def complete_dag_density(d: int) -> float:
    """
    Return the edge density of a complete DAG.

    For d nodes, density = |E| / max_possible_edges = 1.0
    (all possible edges are present)
    """
    if d <= 1:
        return 0.0
    return 1.0
