import pathlib
import networkx as nx
import tsplib95
import os
import subprocess

def create_tsp_problem_object(G, cost="weight"):
    """
    Create a TSPLIB problem object from a networkx graph.

    Parameters
    ----------
    G : networkx.Graph
        The input graph.
    cost : str
        The edge attribute to use as cost.

    Returns
    -------
    problem : tsplib95.models.StandardProblem
        The TSPLIB problem object.
    """
    problem = tsplib95.models.StandardProblem()
    problem.name = "tmp"
    problem.type = "TSP"
    problem.dimension = G.number_of_nodes()
    problem.edge_weight_type = "EXPLICIT"
    problem.edge_weight_format = "UPPER_ROW"
    edge_weight_section = []
    for i in range(problem.dimension):
        for j in range(i + 1, problem.dimension):
            edge_weight_section.append(int(G[i][j][cost]))
    problem.edge_weights = [edge_weight_section]
    return problem

def parse_log(filename):
    """
    Parse the log file generated by Concorde.

    Parameters
    ----------
    filename : str
        The name of the log file.

    Returns
    -------
    ot : float
        The optimal solution value.
    bb_nodes : float
        The number of branch-and-bound nodes.
    time : float
        The total running time.
    """
    F = open(filename)
    lines = F.readlines()
    F.close()
    ot_line = next(filter(lambda x : "Optimal Solution:" in x, list(lines.__reversed__())))
    bb_nodes_line = ""
    try:
        bb_nodes_line = next(filter(lambda x : "Number of bbnodes:" in x, list(lines.__reversed__())))
    except:
        pass
    time_line = next(filter(lambda x : "Total Running Time:" in x , list(lines.__reversed__())))
    ot = float(ot_line.split(":")[1].strip())
    bb_nodes = 0
    if len(bb_nodes_line) >= 1:
        bb_nodes = float(bb_nodes_line.split(":")[1].strip())
    time = float(time_line.split(":")[1].split("(")[0])
    return ot, bb_nodes, time

def parse_sol(filename, as_tour = False, as_list_of_edges = False):
    """
    Parse the solution file generated by Concorde.

    Parameters
    ----------
    filename : str
        The name of the solution file.
    as_tour : bool
        If True, return the tour as a list of nodes.
    as_list_of_edges : bool
        If True, return the tour as a list of edges.

    Returns
    -------
    X : list
        The tour as a list of nodes or edges. If it is a list of nodes, it ends with the first node, that is most likely 0
    """
    # Just one between as_tour and as_list_of_edges must be True
    assert as_tour != as_list_of_edges, "Only one of as_tour and as_list_of_edges must be True"

    # Open the sol file
    F = open(filename, "r")

    # Skip the first line
    F.readline()

    # Read the other lines
    lines = F.readlines()

    nodes = []
    for line in lines:
        line_vec = line.strip().split(" ")
        for u in line_vec:
            if len(u) > 0:
                nodes.append(int(u))

    nodes += [nodes[0]]
    if as_tour:
        return nodes

    else:
        edges = []
        for i in range(len(nodes) - 1):
            u = min(nodes[i], nodes[i + 1])
            v = max(nodes[i], nodes[i + 1])
            edges.append((u, v))
        return edges

def run_concorde(G, cost=None, concorde_path=None, seed = None, options=[], verbose=False, remove_all=False, get_tour = False, get_edges = False):
    """
    Run the Concorde TSP solver on a given graph.

    Parameters
    ----------
    G : networkx.Graph or str
        The input graph or the path to a TSPLIB file.
    cost : str, optional
        The edge attribute to use as cost. If None, the graph must have 'coord' attribute for each node.
    concorde_path : str
        The path to the Concorde executable.
    seed : int, optional
        The random seed to use.
    options : list of str, optional
        Additional command line options to pass to Concorde
    verbose : bool, optional
        If True, print the Concorde output.
    remove_all : bool, optional
        If True, remove all temporary files after execution.
    get_tour : bool, optional
        If True, return the optimal tour as a list of nodes.
    get_edges : bool, optional
        If True, return the optimal tour as a list of edges.

    Returns
    -------
    ot : float
        The optimal solution value.
    bb_nodes : float
        The number of branch-and-bound nodes.
    time : float
        The total running time.
    X_t : list, optional
        The optimal tour as a list of nodes (if get_tour is True).
    X_e : list, optional
        The optimal tour as a list of edges (if get_edges is True).
    """
    assert concorde_path != None, "Concorde path must be provided"


    concorde_path = pathlib.Path(concorde_path)

    # Create a directory named tmp to store temporary files
    tmp_dir = "./tmp/"
    pathlib.Path(tmp_dir).mkdir(parents=True, exist_ok=True)

    if type(G) == nx.Graph:
        # Check if the cost is equal to None
        if cost is None:
            # Then, you need coordinate attributes
            if "coord" not in G.nodes[0]:
                raise ValueError("If cost is None, the graph must have 'coord' attribute for each node; else, provide the cost attribute to use")
            tsp_file = tmp_dir + "tmp.tsp"
            to_write = f"NAME: tmp\nTYPE: TSP\nDIMENSION: {G.number_of_nodes()}\nEDGE_WEIGHT_TYPE: EUC_2D\nNODE_COORD_SECTION\n"
            for i in range(G.number_of_nodes()):
                x, y = G.nodes[i]["coord"]
                to_write += f" {i + 1} {x} {y}\n"
            to_write += "EOF\n"
            with open(tsp_file, 'w') as f:
                f.write(to_write)
            # Now you are ready to run concorde
        else:
            problem = create_tsp_problem_object(G, cost=cost)
            tsp_file = tmp_dir + "tmp.tsp"
            with open(tsp_file, 'w') as f:
                problem.write(f)
    else:
        assert type(G) == str, "G must be a networkx.Graph or a string representing the path to a TSPLIB file"
        F = open(G, "r")
        lines = F.read()
        F.close()
        F = open(tmp_dir + "tmp.tsp", "w+")
        F.write(lines)
        F.close()

    # Change the directory
    os.chdir(tmp_dir)

    # ⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐
    # Run Concorde
    # ⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐⭐

    # is verbose?
    if not verbose:
        options += ["-x"]
    # Do we want to use the seed?
    if seed != None:
        options += ["-s {}".format(seed)]

    # Merge all the options together
    options = " ".join(options)

    # ~/libraries/concorde/build/TSP/concorde tmp.tsp > tmp.log
    subprocess.run(f"{concorde_path} {options} tmp.tsp > tmp.log", shell=True)
    ot, bb_nodes, time = parse_log("tmp.log")

    # Now, if you also want the tour
    if get_tour:
        # Open the sol file
        X_t = parse_sol("tmp.sol", as_tour=True)
    if get_edges:
        X_e = parse_sol("tmp.sol", as_list_of_edges=True)

    # Return to the original directory
    os.chdir("..")

    if remove_all:
        # Remove the tmp directory
        os.rmdir(tmp_dir, recursive=True)
    if get_tour and not get_edges:
        return ot, bb_nodes, time, X_t
    if get_edges and not get_tour:
        return ot, bb_nodes, time, X_e
    if get_tour and get_edges:
        return ot, bb_nodes, time, X_t, X_e

    # Just return the optimal value and the bb nodes
    return ot, bb_nodes, time
