{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "c78646a5-1243-4cf2-9ff0-be144d5743b3",
   "metadata": {},
   "outputs": [],
   "source": [
    "import time\n",
    "import pandas as pd\n",
    "from tqdm import tqdm\n",
    "import numpy as np\n",
    "import time\n",
    "import matplotlib.pyplot as plt\n",
    "from tqdm import trange\n",
    "from tsp_rtdl_fast import plot_tour, two_opt_rtdl, two_opt, Logger, calculate_total_distance\n",
    "from tsp_rtdl_util import create_problem\n",
    "from TSPlib_utils import create_problem_from_dict, plot_multiple_tours\n",
    "import torch\n",
    "import scipy\n",
    "\n",
    "import tsplib95\n",
    "import scipy.spatial\n",
    "import requests, tarfile, os, gzip, shutil\n",
    "from tqdm.auto import tqdm\n",
    "from tsplib95.loaders import load_problem, load_solution\n",
    "import os\n",
    "\n",
    "from concorde import *"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "769ed864-5650-46a0-b78b-9fbbeed33894",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Load the problem from TSPLib\n",
    "tsplib_dir = './tsplib'# modify this to the directory of your prepared files\n",
    "files = os.listdir(tsplib_dir)\n",
    "problem_files_full = [file for file in files if file.endswith('.tsp')]\n",
    "\n",
    "# Load the optimal solution files from TSPLib\n",
    "solution_files = [file for file in files if file.endswith('.opt.tour')] \n",
    "\n",
    "problems = []\n",
    "# Load only problems with solution files\n",
    "for sol_file in solution_files:\n",
    "    prob_file = sol_file.replace('.opt.tour', '.tsp')\n",
    "    problem = load_problem(os.path.join(tsplib_dir, prob_file))\n",
    "\n",
    "    # NOTE: in some problem files (e.g. hk48), the node coordinates are not available\n",
    "    # we temporarily skip these problems\n",
    "    if not len(problem.node_coords):\n",
    "        continue\n",
    "    \n",
    "    node_coords = torch.tensor([v for v in problem.node_coords.values()])\n",
    "    # solution = load_solution(os.path.join(tsplib_dir, sol_file))\n",
    "    # solution = solve_concorde_log('concorde_build/TSP/concorde', '.', 'test', node_coords)\n",
    "    \n",
    "    problems.append({\n",
    "        \"name\": sol_file.replace('.opt.tour', ''),\n",
    "        \"node_coords\": node_coords,\n",
    "        \"dimension\": problem.dimension\n",
    "    })\n",
    "    \n",
    "    \n",
    "# order by dimension\n",
    "problems = sorted(problems, key=lambda x: x['dimension'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c3d17b1a-02d5-4427-a9c0-3cf4f9b3f6d4",
   "metadata": {},
   "outputs": [],
   "source": [
    "MAX_ITER = 10**4"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6e3894fa-fc61-4b72-a756-4a3dc3618fa2",
   "metadata": {},
   "outputs": [],
   "source": [
    "from scipy.spatial import distance_matrix\n",
    "from tqdm.auto import tqdm\n",
    "\n",
    "def tour_length_from_matrix(tour, distance_matrix_np):\n",
    "    L = 0.0\n",
    "    n = len(tour)\n",
    "    for i in range(n):\n",
    "        a = tour[i]\n",
    "        b = tour[(i + 1) % n]\n",
    "        L += distance_matrix_np[a, b]\n",
    "    return float(L)\n",
    "\n",
    "def mean_std(arr):\n",
    "    arr = np.asarray(arr, dtype=float)\n",
    "    if len(arr) == 0:\n",
    "        return np.nan, np.nan\n",
    "    if len(arr) == 1:\n",
    "        return float(arr[0]), 0.0\n",
    "    return float(np.mean(arr)), float(np.sqrt(np.var(arr)))\n",
    "\n",
    "    index = routing.Start(0)\n",
    "    route = []\n",
    "    while not routing.IsEnd(index):\n",
    "        route.append(manager.IndexToNode(index))\n",
    "        index = solution.Value(routing.NextVar(index))\n",
    "    return route\n",
    "\n",
    "\n",
    "N_RUNS = 1000\n",
    "time_limit = 20\n",
    "rtdl_period = None\n",
    "\n",
    "logger_orig = Logger(MAX_ITER)\n",
    "logger_rtdl = Logger(MAX_ITER)\n",
    "logger_rtdl_progressive = Logger(MAX_ITER)\n",
    "\n",
    "results_rows = []\n",
    "\n",
    "for problem in tqdm(problems[:10]):\n",
    "    cities, distance_matrix_np = create_problem_from_dict(problem)\n",
    "    n = cities.shape[0]\n",
    "    if rtdl_period:\n",
    "        rtdl_period_t = rtdl_period if rtdl_period != 'adaptive' else 2 if n<50 else 4\n",
    "    else:\n",
    "        rtdl_period_t = None\n",
    "    print(\"RTDL period: \", rtdl_period_t)\n",
    "    \n",
    "    base_times, base_dists = [], []\n",
    "    rtdl_times, rtdl_dists = [], []\n",
    "    rtdl_prog_times, rtdl_prog_dists = [], []\n",
    "\n",
    "    best_base = {\"dist\": float(\"inf\"), \"tour\": None}\n",
    "    best_rtdl = {\"dist\": float(\"inf\"), \"tour\": None}\n",
    "    best_rtdl_prog = {\"dist\": float(\"inf\"), \"tour\": None}\n",
    "\n",
    "    for _ in tqdm(range(N_RUNS)):\n",
    "        initial_tour = list(range(cities.shape[0]))\n",
    "        np.random.shuffle(initial_tour)\n",
    "\n",
    "        # --- Two-Opt ---\n",
    "        t0 = time.perf_counter()\n",
    "        logger_orig.problem_started()\n",
    "        tour_base, dist_base = two_opt(\n",
    "            initial_tour, distance_matrix_np, max_iterations=MAX_ITER, logger=logger_orig, time_limit=time_limit\n",
    "        )\n",
    "        logger_orig.problem_solved()\n",
    "        base_time = time.perf_counter() - t0\n",
    "\n",
    "        base_times.append(base_time)\n",
    "        base_dists.append(dist_base)\n",
    "        if dist_base < best_base[\"dist\"]:\n",
    "            best_base.update({\"dist\": dist_base, \"tour\": tour_base})\n",
    "\n",
    "        # --- RTDL (base) ---\n",
    "        t0 = time.perf_counter()\n",
    "        logger_rtdl.problem_started()\n",
    "        tour_rtdl, dist_rtdl = two_opt_rtdl(\n",
    "            initial_tour, distance_matrix_np, max_iterations=MAX_ITER,\n",
    "            logger=logger_rtdl, progressive=False, rtdl_period = rtdl_period_t, time_limit=time_limit\n",
    "        )\n",
    "        logger_rtdl.problem_solved()\n",
    "        rtdl_time = time.perf_counter() - t0\n",
    "\n",
    "        rtdl_times.append(rtdl_time)\n",
    "        rtdl_dists.append(dist_rtdl)\n",
    "        if dist_rtdl < best_rtdl[\"dist\"]:\n",
    "            best_rtdl.update({\"dist\": dist_rtdl, \"tour\": tour_rtdl})\n",
    "\n",
    "        # --- RTDL (progressive) ---\n",
    "        t0 = time.perf_counter()\n",
    "        logger_rtdl_progressive.problem_started()\n",
    "        tour_rtdl_prog, dist_rtdl_prog = two_opt_rtdl(\n",
    "            initial_tour, distance_matrix_np, max_iterations=MAX_ITER,\n",
    "            logger=logger_rtdl_progressive, progressive=True, rtdl_period = rtdl_period_t, time_limit=time_limit\n",
    "        )\n",
    "        logger_rtdl_progressive.problem_solved()\n",
    "        rtdl_prog_time = time.perf_counter() - t0\n",
    "\n",
    "        rtdl_prog_times.append(rtdl_prog_time)\n",
    "        rtdl_prog_dists.append(dist_rtdl_prog)\n",
    "        if dist_rtdl_prog < best_rtdl_prog[\"dist\"]:\n",
    "            best_rtdl_prog.update({\"dist\": dist_rtdl_prog, \"tour\": tour_rtdl_prog})\n",
    "\n",
    "    # try:\n",
    "    print(f\"Calculating Concord for: {problem['name']}\")\n",
    "    t0 = time.perf_counter()\n",
    "    tour_pytsp = solve_concorde_log('concorde_build/TSP/concorde', '.', 'test', None, problem_filename = os.path.join(tsplib_dir, problem['name']+'.tsp'))\n",
    "    concorde_time = time.perf_counter() - t0\n",
    "    pytsp_dist = tour_length_from_matrix(tour_pytsp, distance_matrix_np)\n",
    "\n",
    "    base_time_mean, base_time_std = mean_std(base_times)\n",
    "    base_dist_mean, base_dist_std = mean_std(base_dists)\n",
    "\n",
    "    rtdl_time_mean, rtdl_time_std = mean_std(rtdl_times)\n",
    "    rtdl_dist_mean, rtdl_dist_std = mean_std(rtdl_dists)\n",
    "\n",
    "    rtdl_prog_time_mean, rtdl_prog_time_std = mean_std(rtdl_prog_times)\n",
    "    rtdl_prog_dist_mean, rtdl_prog_dist_std = mean_std(rtdl_prog_dists)\n",
    "\n",
    "    tours = [\n",
    "        [best_base[\"tour\"], f\"{problem['name']} Two-Opt (best)\"],\n",
    "        [best_rtdl[\"tour\"], f\"{problem['name']} RTDL (best)\"],\n",
    "        [best_rtdl_prog[\"tour\"], f\"{problem['name']} RTDL progressive (best)\"],\n",
    "    ]\n",
    "    if tour_pytsp is not None:\n",
    "        tours.append([tour_pytsp, f\"{problem['name']} Concord\"])\n",
    "\n",
    "    plot_multiple_tours(cities, tours)\n",
    "\n",
    "    results_rows.append([\n",
    "        problem[\"name\"],\n",
    "\n",
    "        base_time_mean, base_dist_mean, base_dist_mean/pytsp_dist - 1, base_dist_std,\n",
    ",\n",
    "        rtdl_time_mean, rtdl_dist_mean, rtdl_dist_mean/pytsp_dist -1 , rtdl_dist_std,\n",
    "\n",
    "        rtdl_prog_time_mean, rtdl_prog_dist_mean, rtdl_prog_dist_mean/pytsp_dist -1 , rtdl_prog_dist_std,\n",
    "        \n",
    "        concorde_time, pytsp_dist, 1,\n",
    "    ])\n",
    "\n",
    "\n",
    "columns = pd.MultiIndex.from_tuples([\n",
    "    (\"City\", \"\"),\n",
    "    (\"Two-Opt\", \"Time (mean)\"), (\"Two-Opt\", \"Distance (mean)\"), (\"Two-Opt\", \"GAP (mean)\"), (\"Two-Opt\", \"Distance (std)\"),\n",
    "    (\"RTDL\", \"Time (mean)\"), (\"RTDL\", \"Distance (mean)\"), (\"RTDL\", \"GAP (mean)\"), (\"RTDL\", \"Distance (std)\"),\n",
    "    (\"RTDL Progressive\", \"Time (mean)\"), (\"RTDL Progressive\", \"Distance (mean)\"), (\"RTDL Progressive\", \"GAP (mean)\"), (\"RTDL Progressive\", \"Distance (std)\"),\n",
    "    (\"Concord\", \"time\"), (\"Concord\", \"Distance (mean)\"), (\"Concord\", \"GAP\")\n",
    "])\n",
    "\n",
    "results_df = pd.DataFrame(results_rows, columns=columns)\n",
    "\n",
    "results_df.round(4).to_excel(f\"tsp_results_rtdl_period_last.xlsx\")\n",
    "print(f\"Saved: tsp_results_rtdl_period_last.xlsx\")\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.10.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
