"""Parallel Insertion heuristic for CVRP."""

from __future__ import annotations

from typing import List, Tuple, Optional
import numpy as np

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


def compute_insertion_cost(
    route: List[int],
    customer: int,
    position: int,
    distance_matrix: np.ndarray
) -> float:
    """
    Compute cost of inserting customer at given position in route.
    
    Args:
        route: Current route (starts and ends at depot 0)
        customer: Customer node ID to insert
        position: Position to insert (1-based, between existing nodes)
        distance_matrix: Distance matrix
        
    Returns:
        Additional cost of insertion
    """
    if position < 1 or position >= len(route):
        return float('inf')
    
    # Cost = d(prev, customer) + d(customer, next) - d(prev, next)
    prev_node = route[position - 1]
    next_node = route[position]
    
    cost = (
        distance_matrix[prev_node, customer] +
        distance_matrix[customer, next_node] -
        distance_matrix[prev_node, next_node]
    )
    
    return cost


def find_best_insertion(
    route: List[int],
    customer: int,
    demands: np.ndarray,
    vehicle_capacity: float,
    distance_matrix: np.ndarray
) -> Tuple[Optional[int], float]:
    """
    Find best position to insert customer in route.
    
    Args:
        route: Current route
        customer: Customer node ID to insert
        demands: Demands array
        vehicle_capacity: Vehicle capacity
        distance_matrix: Distance matrix
        
    Returns:
        (best_position, best_cost) or (None, inf) if not feasible
    """
    # Check capacity constraint
    current_demand = sum(demands[node] for node in route if node != 0)
    if current_demand + demands[customer] > vehicle_capacity:
        return None, float('inf')
    
    # Try all positions (excluding depot endpoints)
    best_position = None
    best_cost = float('inf')
    
    for pos in range(1, len(route)):
        cost = compute_insertion_cost(route, customer, pos, distance_matrix)
        if cost < best_cost:
            best_cost = cost
            best_position = pos
    
    return best_position, best_cost


def solve(instance: dict) -> BaselineResult:
    """
    Solve CVRP instance using Parallel Insertion heuristic.
    
    Algorithm:
    1. Start with multiple empty routes (one per vehicle or based on demand)
    2. For each customer, find best insertion across all routes
    3. Insert customer at best position
    4. Routes are built in parallel (simultaneously)
    
    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)
    
    # Estimate number of routes needed (lower bound)
    total_demand = sum(demands[i] for i in range(1, n_plus_1))
    min_routes = max(1, int(np.ceil(total_demand / vehicle_capacity)))
    
    # Initialize: start with multiple empty routes
    routes = [[0, 0] for _ in range(min_routes)]
    
    # Sort customers by some criterion (e.g., distance from depot)
    unvisited = list(range(1, n_plus_1))
    depot_distances = [(distance_matrix[0, i], i) for i in unvisited]
    depot_distances.sort()
    unvisited = [i for _, i in depot_distances]
    
    # Insert customers in parallel (all routes available simultaneously)
    for customer in unvisited:
        best_route_idx = None
        best_position = None
        best_cost = float('inf')
        
        # Try inserting in all existing routes
        for route_idx, route in enumerate(routes):
            pos, cost = find_best_insertion(
                route, customer, demands, vehicle_capacity, distance_matrix
            )
            if pos is not None and cost < best_cost:
                best_cost = cost
                best_position = pos
                best_route_idx = route_idx
        
        # Insert customer
        if best_route_idx is not None:
            # Insert into existing route
            routes[best_route_idx].insert(best_position, customer)
        else:
            # Create new route
            new_route = [0, customer, 0]
            routes.append(new_route)
    
    # Remove empty routes
    routes = [r for r in routes if len(r) > 2]  # More than just [0, 0]
    
    # If no routes created (shouldn't happen), create empty route
    if not routes:
        routes = [[0]]
    
    # Compute total cost
    cost = compute_routes_cost(routes, distance_matrix)
    
    return BaselineResult(routes=routes, cost=cost)

