from typing import List, Tuple, Set, Dict
import random, statistics

random.seed(42)

def scale_to_degrees(weights: List[float], target_avg_deg: float = 6.0) -> List[int]:
    n = len(weights)
    S = int(round(n * target_avg_deg))
    if S % 2 == 1:
        S += 1
    wsum = sum(weights)
    c = S / wsum
    deg = [max(1, min(n-1, int(round(c * w)))) for w in weights]
    if sum(deg) % 2 == 1:
        i = max(range(n), key=lambda i: (n-1-deg[i], weights[i]))
        if deg[i] < n-1: deg[i] += 1
        else:
            j = max(range(n), key=lambda j: (deg[j]-1, weights[j]))
            deg[j] -= 1
    return deg


def havel_hakimi_construct_(deg: List[int]) -> Dict[int, Set[int]]:
    n = len(deg)
    adj: Dict[int, Set[int]] = {i: set() for i in range(n)}

    rem = [[deg[i], i] for i in range(n)]

    while True:
        rem.sort(key=lambda x: x[0], reverse=True)

        if rem[0][0] == 0:
            return adj

        r, u = rem.pop(0)

        if r > len(rem):
            raise ValueError(f"Graph realization failed: Not enough nodes remaining for node {u} with degree {r}")

        candidates_indices = []

        for i, (d_v, v) in enumerate(rem):
            if v not in adj[u]:
                candidates_indices.append(i)
                if len(candidates_indices) == r:
                    break

        if len(candidates_indices) < r:
            raise ValueError(
                f"Graph realization failed: Cannot find {r} non-neighbor vertices for node {u} to maintain simplicity.")

        for idx in candidates_indices:
            d_v, v = rem[idx]

            adj[u].add(v)
            adj[v].add(u)
            rem[idx][0] -= 1
            if rem[idx][0] < 0:
                raise ValueError("Graph realization failed: Negative degree detected.")

    return adj
def havel_hakimi_construct(deg: List[int]) -> Dict[int, Set[int]]:
    n = len(deg)
    rem = [(deg[i], i) for i in range(n)]
    adj: Dict[int, Set[int]] = {i: set() for i in range(n)}
    if sum(d for d, _ in rem) % 2: raise ValueError("odd sum")
    max_retries = n * 5
    retry_count = 0
    while True:
        rem.sort(reverse=True)
        if rem[0][0] == 0: return adj
        d, u = rem[0]; rem = rem[1:]
        if d > len(rem): raise ValueError("not graphical")
        for i in range(d):
            dv, v = rem[i]
            if v == u or v in adj[u]:
                swapped = False
                for j in range(d, len(rem)):
                    dv2, v2 = rem[j]
                    if v2 != u and v2 not in adj[u]:
                        rem[i], rem[j] = rem[j], rem[i]
                        dv, v = rem[i]
                        swapped = True
                        break

                if not swapped:
                    retry_count += 1
                    if retry_count > max_retries:
                        raise ValueError("Too many retries in construction")
                    continue
            adj[u].add(v); adj[v].add(u)
            rem[i] = (dv-1, v)
            if rem[i][0] < 0: raise ValueError("negative")

def connected_components(n: int, adj: Dict[int, Set[int]]):
    seen = [False]*n; comps = []
    for s in range(n):
        if not seen[s]:
            st=[s]; seen[s]=True; comp=[]
            while st:
                u=st.pop(); comp.append(u)
                for v in adj[u]:
                    if not seen[v]: seen[v]=True; st.append(v)
            comps.append(comp)
    return comps

def try_edge_swap_connect(n: int, adj: Dict[int, Set[int]]) -> bool:
    comps = connected_components(n, adj)
    if len(comps) <= 1: return False
    c1, c2 = comps[0], comps[1]
    import random
    e1 = [(u,v) for u in c1 for v in adj[u] if u<v and v in c1]
    e2 = [(u,v) for u in c2 for v in adj[u] if u<v and v in c2]
    random.shuffle(e1); random.shuffle(e2)
    for (a,b) in e1:
        for (c,d) in e2:
            for (x,y),(p,q) in [((a,c),(b,d)), ((a,d),(b,c))]:
                if x==y or p==q: continue
                if y in adj[x] or q in adj[p]: continue
                adj[a].remove(b); adj[b].remove(a)
                adj[c].remove(d); adj[d].remove(c)
                adj[x].add(y); adj[y].add(x)
                adj[p].add(q); adj[q].add(p)
                return True
    return False

def make_connected(n: int, adj: Dict[int, Set[int]]) -> Dict[int, Set[int]]:
    for _ in range(1000):
        if len(connected_components(n, adj)) == 1: return adj
        if not try_edge_swap_connect(n, adj): break
    return adj


def build_graph_from_weights(
        weights: List[float],
        target_avg_deg: float = 6.1
) -> Tuple[List[int], Dict[int, Set[int]]]:
    n = len(weights)
    for d_avg in [target_avg_deg, 6.0, 6.5, 5.5, 7.0, 5.0]:
        deg = scale_to_degrees(weights, d_avg)
        for _ in range(20):
            try:
                adj = havel_hakimi_construct(deg)
                adj = make_connected(n, adj)
                if all(len(adj[i]) == deg[i] for i in range(n)) and len(connected_components(n, adj))==1:
                    return deg, adj
            except ValueError:
                i = max(range(n), key=lambda k: deg[k])
                j = min(range(n), key=lambda k: deg[k])
                if deg[i] > 1 and deg[j] < n-1:
                    deg[i] -= 1; deg[j] += 1
                if sum(deg) % 2:
                    if deg[j] < n-1: deg[j] += 1
                    else: deg[i] -= 1
    deg = scale_to_degrees(weights, 6.0)
    adj = {i:set() for i in range(n)}
    for i in range(n):
        u,v = i,(i+1)%n
        adj[u].add(v); adj[v].add(u)
    need = [deg[i]-2 for i in range(n)]
    nodes = list(range(n))
    for _ in range(n*(n-1)//2):
        nodes.sort(key=lambda i: need[i], reverse=True)
        u = nodes[0]
        if need[u] <= 0: break
        for v in nodes[1:]:
            if need[v] <= 0: continue
            if v not in adj[u]:
                adj[u].add(v); adj[v].add(u)
                need[u] -= 1; need[v] -= 1
                break
    return deg, make_connected(n, adj)

def adjacency_matrix(adj: Dict[int, Set[int]]):
    n = len(adj)
    A = [[0]*n for _ in range(n)]
    for u in range(n):
        for v in adj[u]:
            A[u][v] = 1; A[v][u] = 1
    return A
