"""Nearest Neighbor + 2-opt heuristic for CVRP."""

from __future__ import annotations

from typing import List
import numpy as np

from ..utils import BaselineResult, compute_routes_cost, two_opt_improve
from heupsro.problems.cvrp.evolution.shared.solver.solve_instance import instance_to_solver_inputs


def solve(instance: dict) -> BaselineResult:
    """
    Solve CVRP instance using Nearest Neighbor heuristic with 2-opt improvement.
    
    Algorithm:
    1. Use nearest neighbor to construct routes
    2. Apply 2-opt improvement to each route
    
    Args:
        instance: CVRP instance dict with 'depot', 'customers', 'vehicle_capacity'
        
    Returns:
        BaselineResult with routes and cost
    """
    # Convert instance to solver inputs
    distance_matrix, demands, vehicle_capacity = instance_to_solver_inputs(instance)
    
    n_plus_1 = len(demands)
    n_customers = n_plus_1 - 1
    
    if n_customers == 0:
        return BaselineResult(routes=[[0]], cost=0.0)
    
    # Step 1: Construct routes using nearest neighbor
    routes = []
    unvisited = set(range(1, n_plus_1))
    
    while unvisited:
        # Start new route from depot
        route = [0]
        current_node = 0
        current_load = 0.0
        
        # Build route using nearest neighbor
        while unvisited:
            # Find feasible unvisited customers
            feasible = [u for u in unvisited if demands[u] <= vehicle_capacity - current_load]
            
            if not feasible:
                break
            
            # Find nearest feasible customer
            nearest = None
            min_dist = float('inf')
            
            for node in feasible:
                dist = distance_matrix[current_node, node]
                if dist < min_dist:
                    min_dist = dist
                    nearest = node
            
            if nearest is None:
                break
            
            # Add to route
            route.append(nearest)
            current_load += demands[nearest]
            unvisited.remove(nearest)
            current_node = nearest
        
        # Return to depot
        route.append(0)
        routes.append(route)
    
    # Step 2: Apply 2-opt improvement to each route
    improved_routes = []
    for route in routes:
        improved_route = two_opt_improve(route, distance_matrix, max_iterations=100)
        improved_routes.append(improved_route)
    
    # Compute total cost
    cost = compute_routes_cost(improved_routes, distance_matrix)
    
    return BaselineResult(routes=improved_routes, cost=cost)

