import numpy as np
from typing import Optional

def static_exponential_graph(n: int) -> np.ndarray:
    A = np.zeros((n, n), dtype=np.int64)
    p = 0
    while (1 << p) <= n // 2:
        d = 1 << p
        for i in range(n):
            j = (i + d) % n
            A[i, j] = 1
            A[j, i] = 1
        p += 1

    return A

def complete_graph(n: int) -> np.ndarray:
    A = np.ones((n, n), dtype=np.int64)
    np.fill_diagonal(A, 0)
    return A


def ring_graph(n: int, k: int = 1) -> np.ndarray:
    A = np.zeros((n, n), dtype=np.int64)
    for i in range(n):
        for t in range(1, k + 1):
            A[i, (i - t) % n] = 1
            A[i, (i + t) % n] = 1
    return A


def star_graph(n: int, center: int = 0) -> np.ndarray:
    A = np.zeros((n, n), dtype=np.int64)
    for i in range(n):
        if i == center:
            continue
        A[center, i] = 1
        A[i, center] = 1
    return A


def erdos_renyi_graph(n: int, p: float, seed: Optional[int] = None) -> np.ndarray:

    if n < 0:
        raise ValueError("n must be non-negative")
    if not (0.0 <= p <= 1.0):
        raise ValueError("p must be in [0, 1]")

    rng = np.random.default_rng(seed)
    A = np.zeros((n, n), dtype=np.int64)
    if n <= 1 or p == 0.0:
        return A
    if p == 1.0:
        A[:] = 1
        np.fill_diagonal(A, 0)
        return A

    R = rng.random((n, n))
    U = (np.triu(R, k=1) < p).astype(np.int64)
    A = U + U.T
    return A


def grid_2d_graph(m: int, n: int, connectivity: int = 4) -> np.ndarray:
    assert connectivity in (4, 8)
    N = m * n
    A = np.zeros((N, N), dtype=np.int64)

    def idx(r, c): return r * n + c

    for r in range(m):
        for c in range(n):
            u = idx(r, c)
            for dr, dc in [(-1, 0), (1, 0), (0, -1), (0, 1)]:
                rr, cc = r + dr, c + dc
                if 0 <= rr < m and 0 <= cc < n:
                    v = idx(rr, cc)
                    A[u, v] = 1
                    A[v, u] = 1
            if connectivity == 8:
                for dr, dc in [(-1, -1), (-1, 1), (1, -1), (1, 1)]:
                    rr, cc = r + dr, c + dc
                    if 0 <= rr < m and 0 <= cc < n:
                        v = idx(rr, cc)
                        A[u, v] = 1
                        A[v, u] = 1
    return A


def random_geometric_graph(n: int, radius: float, seed: Optional[int] = None) -> np.ndarray:
    if n <= 0:
        raise ValueError("n must be positive")
    if radius <= 0.0:
        raise ValueError("radius must be positive")

    rng = np.random.default_rng(seed)
    pos = rng.random((n, 2))

    def build_adj(r):
        diff = pos[:, None, :] - pos[None, :, :]
        dist = np.sqrt(np.sum(diff ** 2, axis=-1))
        A = (dist <= r).astype(np.int64)
        np.fill_diagonal(A, 0)
        return A

    def is_connected(A):
        visited = np.zeros(n, dtype=bool)
        stack = [0]
        while stack:
            i = stack.pop()
            if not visited[i]:
                visited[i] = True
                neighbors = np.where(A[i] > 0)[0]
                stack.extend(neighbors)
        return visited.all()

    A = build_adj(radius)
    while not is_connected(A):
        radius *= 1.1
        A = build_adj(radius)
        if radius >= np.sqrt(2):
            break

    return A