"""
Star graph generator.

A star graph has a central node connected to all other nodes.
"""

from typing import Optional
from ..core.dag import DAG


def generate_star(
    d: int,
    center: int = 0,
    center_is_parent: bool = True
) -> DAG:
    """
    Generate a star DAG with specified center.

    Args:
        d: Number of nodes (center + leaves)
        center: Index of the center node (default: 0)
        center_is_parent: If True, center -> leaves; if False, leaves -> center

    Returns:
        Star DAG with d-1 edges

    Properties:
        - If center_is_parent: No v-structures, MEC size = d+1
        - If center_is_child: (d-1 choose 2) v-structures, MEC size = 1
    """
    if d < 1:
        raise ValueError(f"Number of nodes must be positive, got {d}")
    if not 0 <= center < d:
        raise ValueError(f"Center {center} out of bounds for d={d}")

    dag = DAG(d)

    if d == 1:
        return dag  # Single node, no edges

    leaves = [i for i in range(d) if i != center]

    if center_is_parent:
        for leaf in leaves:
            dag.add_edge(center, leaf)
    else:
        for leaf in leaves:
            dag.add_edge(leaf, center)

    return dag


def generate_star_outward(d: int, center: int = 0) -> DAG:
    """
    Generate a star DAG with center as parent (outward edges).

    center -> leaf_1, center -> leaf_2, ..., center -> leaf_{d-1}

    Properties:
        - No v-structures
        - All leaves independent given center
    """
    return generate_star(d, center=center, center_is_parent=True)


def generate_star_inward(d: int, center: int = 0) -> DAG:
    """
    Generate a star DAG with center as child (inward edges).

    leaf_1 -> center, leaf_2 -> center, ..., leaf_{d-1} -> center

    Properties:
        - (d-1 choose 2) v-structures (all pairs of leaves form v-structure at center)
        - MEC is singleton (all edges compelled)
    """
    return generate_star(d, center=center, center_is_parent=False)


def generate_random_star(d: int, random_state: Optional[int] = None) -> DAG:
    """
    Generate a star DAG with random center and direction.

    Args:
        d: Number of nodes
        random_state: Random seed

    Returns:
        Star DAG with random center and random edge direction
    """
    import numpy as np

    if d < 1:
        raise ValueError(f"Number of nodes must be positive, got {d}")

    rng = np.random.default_rng(random_state)

    center = rng.integers(0, d)
    center_is_parent = rng.random() < 0.5

    return generate_star(d, center=center, center_is_parent=center_is_parent)


def count_star_v_structures(d: int, center_is_parent: bool) -> int:
    """
    Count the number of v-structures in a star graph.

    Args:
        d: Number of nodes
        center_is_parent: Whether center is parent of leaves

    Returns:
        Number of v-structures
    """
    if d <= 2:
        return 0

    if center_is_parent:
        return 0
    else:
        # All pairs of leaves form v-structure at center
        # (d-1 choose 2) = (d-1)(d-2)/2
        return (d - 1) * (d - 2) // 2
