
import time
import importlib
import numpy as np

from ..utils import utils
from . import gls_evol

def solve_instance(n,opt_cost,dis_matrix,coord,time_limit, ite_max, perturbation_moves,heuristic):

    t = time.time()

    try:
        init_tour = gls_evol.nearest_neighbor_2End(dis_matrix, 0).astype(int)
        init_cost = utils.tour_cost_2End(dis_matrix, init_tour)
        nb = 100
        nearest_indices = np.argsort(dis_matrix, axis=1)[:, 1:nb+1].astype(int)

        gls_start_time = time.time()
        best_tour, best_cost, iter_i = gls_evol.guided_local_search(coord, dis_matrix, nearest_indices, init_tour, init_cost,
                                                        t + time_limit, ite_max, perturbation_moves,
                                                        first_improvement=False, guide_algorithm=heuristic)
        gls_time = time.time() - gls_start_time
        
        try:
            from ......utils.evaluation_timer import get_timer
            get_timer().add_gls_solver_time(gls_time)
        except Exception as e:
            pass

        # If opt_cost is NaN or inf, return the actual cost instead of gap
        if np.isnan(opt_cost) or np.isinf(opt_cost) or opt_cost <= 0:
            result = best_cost
        else:
            gap = (best_cost / opt_cost - 1) * 100
            result = gap

    except Exception as e:
        if np.isnan(opt_cost):
            result = 1E10  
        else:
            result = 1E10 
    
    return result