"""
Erdős-Rényi random DAG generator.

Generates random DAGs G(d, p) where each possible edge
is included independently with probability p.
"""

from typing import Optional, List, Tuple
import numpy as np

from ..core.dag import DAG


def generate_erdos_renyi_dag(
    d: int,
    p: float,
    random_state: Optional[int] = None,
    ordering: Optional[List[int]] = None
) -> DAG:
    """
    Generate an Erdős-Rényi random DAG G(d, p).

    Algorithm:
    1. Generate a random topological ordering (or use provided one)
    2. For each ordered pair (i, j) with i before j:
       Add edge i -> j with probability p

    Args:
        d: Number of nodes
        p: Edge probability (0 < p < 1)
        random_state: Random seed
        ordering: Topological ordering (default: random permutation)

    Returns:
        Random DAG with expected p * d(d-1)/2 edges

    Properties:
        - Expected number of edges: p * d(d-1)/2
        - V-structure pattern depends on random edges
    """
    if d < 1:
        raise ValueError(f"Number of nodes must be positive, got {d}")
    if not 0 <= p <= 1:
        raise ValueError(f"Edge probability must be in [0, 1], got {p}")

    rng = np.random.default_rng(random_state)

    # Generate or validate ordering
    if ordering is None:
        ordering = list(rng.permutation(d))
    else:
        if len(ordering) != d or set(ordering) != set(range(d)):
            raise ValueError("Invalid ordering")

    dag = DAG(d)

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

    # Add edges with probability p
    for i in range(d):
        for j in range(d):
            if i != j and position[i] < position[j]:
                if rng.random() < p:
                    dag.add_edge(i, j)

    return dag


def generate_erdos_renyi_dag_expected_edges(
    d: int,
    expected_edges: int,
    random_state: Optional[int] = None
) -> DAG:
    """
    Generate ER-DAG with specified expected number of edges.

    Computes p = expected_edges / (d(d-1)/2) and generates G(d, p).

    Args:
        d: Number of nodes
        expected_edges: Expected number of edges
        random_state: Random seed

    Returns:
        Random DAG with approximately expected_edges edges
    """
    max_edges = d * (d - 1) // 2
    if expected_edges < 0 or expected_edges > max_edges:
        raise ValueError(f"Expected edges must be in [0, {max_edges}]")

    p = expected_edges / max_edges if max_edges > 0 else 0
    return generate_erdos_renyi_dag(d, p, random_state)


def generate_erdos_renyi_dag_density(
    d: int,
    density: float,
    random_state: Optional[int] = None
) -> DAG:
    """
    Generate ER-DAG with specified edge density.

    Density = |E| / (d(d-1)/2), so we use p = density.

    Args:
        d: Number of nodes
        density: Edge density in [0, 1]
        random_state: Random seed

    Returns:
        Random DAG with edge density approximately equal to specified density
    """
    if not 0 <= density <= 1:
        raise ValueError(f"Density must be in [0, 1], got {density}")

    return generate_erdos_renyi_dag(d, density, random_state)


def generate_sparse_dag(
    d: int,
    random_state: Optional[int] = None
) -> DAG:
    """
    Generate a sparse random DAG (p = 0.1).

    Args:
        d: Number of nodes
        random_state: Random seed

    Returns:
        Sparse random DAG
    """
    return generate_erdos_renyi_dag(d, p=0.1, random_state=random_state)


def generate_moderate_dag(
    d: int,
    random_state: Optional[int] = None
) -> DAG:
    """
    Generate a moderately dense random DAG (p = 0.2).

    Args:
        d: Number of nodes
        random_state: Random seed

    Returns:
        Moderate density random DAG
    """
    return generate_erdos_renyi_dag(d, p=0.2, random_state=random_state)


def generate_dense_dag(
    d: int,
    random_state: Optional[int] = None
) -> DAG:
    """
    Generate a dense random DAG (p = 0.5).

    Args:
        d: Number of nodes
        random_state: Random seed

    Returns:
        Dense random DAG
    """
    return generate_erdos_renyi_dag(d, p=0.5, random_state=random_state)


def expected_edges(d: int, p: float) -> float:
    """Compute expected number of edges for G(d, p)."""
    return p * d * (d - 1) / 2


def expected_v_structures(d: int, p: float) -> float:
    """
    Compute expected number of v-structures for G(d, p).

    A v-structure i -> k <- j requires:
    - Edge (i, k) present with prob p
    - Edge (j, k) present with prob p
    - Edge (i, j) absent with prob (1-p)

    Number of potential v-structures: d * (d-1 choose 2) positions
    Each has probability: p^2 * (1-p)

    Note: This is approximate as it doesn't account for ordering constraints.
    """
    if d < 3:
        return 0.0

    # For each node k, pairs of potential parents
    # Number of ways to choose k and two other nodes i, j
    num_potential = d * (d - 1) * (d - 2) / 2

    # Probability of v-structure (approximate)
    # More precise calculation would need to account for ordering
    prob_v = p * p * (1 - p) / 2  # Factor of 1/2 for ordering constraints

    return num_potential * prob_v


def edge_density(dag: DAG) -> float:
    """
    Compute the edge density of a DAG.

    Density = |E| / (d(d-1)/2)
    """
    d = dag.num_nodes()
    max_edges = d * (d - 1) // 2
    if max_edges == 0:
        return 0.0
    return dag.num_edges() / max_edges


def generate_multiple_erdos_renyi(
    d: int,
    p: float,
    n_graphs: int,
    random_state: Optional[int] = None
) -> List[DAG]:
    """
    Generate multiple independent ER-DAGs.

    Args:
        d: Number of nodes per graph
        p: Edge probability
        n_graphs: Number of graphs to generate
        random_state: Random seed

    Returns:
        List of n_graphs independent random DAGs
    """
    rng = np.random.default_rng(random_state)

    dags = []
    for _ in range(n_graphs):
        seed = rng.integers(0, 2**31)
        dags.append(generate_erdos_renyi_dag(d, p, random_state=seed))

    return dags
