from __future__ import annotations

import numpy as np
from typing import List


def coords_to_matrix(coords: np.ndarray) -> np.ndarray:
    n = len(coords)
    m = np.zeros((n, n))
    for i in range(n):
        for j in range(n):
            m[i, j] = np.linalg.norm(coords[i] - coords[j])
    return m


def nn_upper_bound(coords: np.ndarray) -> float:
    n = len(coords)
    visited = [False] * n
    order = [0]
    visited[0] = True
    cur = 0
    total = 0.0
    for _ in range(n - 1):
        dmin = 1e18
        nxt = -1
        for j in range(n):
            if not visited[j]:
                d = float(np.linalg.norm(coords[cur] - coords[j]))
                if d < dmin:
                    dmin, nxt = d, j
        total += dmin
        order.append(nxt)
        visited[nxt] = True
        cur = nxt
    total += float(np.linalg.norm(coords[cur] - coords[0]))
    return total


def tour_length(coords: np.ndarray, tour: np.ndarray) -> float:
    total = 0.0
    for i in range(len(tour) - 1):
        total += float(np.linalg.norm(coords[tour[i]] - coords[tour[i + 1]]))
    total += float(np.linalg.norm(coords[tour[-1]] - coords[tour[0]]))
    return total


def mean_gap_identity(instances: List[np.ndarray]) -> float:
    gaps = []
    for coords in instances:
        n = len(coords)
        tour = np.arange(n)
        heur = tour_length(coords, tour)
        opt_est = nn_upper_bound(coords) * 0.6
        gaps.append((heur / opt_est - 1.0) * 100.0)
    return float(np.mean(gaps)) if gaps else 0.0


# --------- simple GLS-like local search (2-opt) utilities ---------
def nearest_neighbor_tour(dist: np.ndarray) -> np.ndarray:
    n = dist.shape[0]
    visited = np.zeros(n, dtype=bool)
    tour = [0]
    visited[0] = True
    cur = 0
    for _ in range(n - 1):
        nxt = -1
        best = 1e18
        for j in range(n):
            if not visited[j] and dist[cur, j] < best:
                best = dist[cur, j]
                nxt = j
        tour.append(nxt)
        visited[nxt] = True
        cur = nxt
    return np.array(tour, dtype=int)


def two_opt_improve(dist: np.ndarray, tour: np.ndarray, max_iters: int = 200) -> np.ndarray:
    n = len(tour)
    def seg_len(a: int, b: int) -> float:
        return dist[a, b]
    improved = True
    it = 0
    while improved and it < max_iters:
        improved = False
        it += 1
        for i in range(1, n - 2):
            for k in range(i + 1, n - 1):
                a, b = tour[i - 1], tour[i]
                c, d = tour[k], tour[k + 1]
                delta = (seg_len(a, c) + seg_len(b, d)) - (seg_len(a, b) + seg_len(c, d))
                if delta < -1e-12:
                    tour[i:k + 1] = tour[i:k + 1][::-1]
                    improved = True
        
    return tour


def solve_tour_by_2opt(dist: np.ndarray) -> np.ndarray:
    tour = nearest_neighbor_tour(dist)
    tour = two_opt_improve(dist, tour)
    return tour


