{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "c78646a5-1243-4cf2-9ff0-be144d5743b3",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import torch\n",
    "import numpy as np\n",
    "import scipy\n",
    "from rtdl import RTD_Lite\n",
    "import matplotlib.pyplot as plt\n",
    "from tqdm import trange\n",
    "from copy import copy\n",
    "import time\n",
    "from tsp_rtdl_util import create_problem, create_problem_nonmetric\n",
    "from subprocess import check_call, check_output, CalledProcessError"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "4cd18a64-408c-4c19-a8ca-4aea6fa3b78f",
   "metadata": {},
   "outputs": [],
   "source": [
    "def calculate_total_distance(tour, distance_matrix):\n",
    "    \"\"\"Calculate the total distance of a tour\"\"\"\n",
    "    total = 0\n",
    "    num_cities = len(tour)\n",
    "    for i in range(num_cities):\n",
    "        total += distance_matrix[tour[i-1], tour[i]]\n",
    "\n",
    "    return total"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "776ac2d4-9b25-4544-888c-38c0186e1993",
   "metadata": {},
   "outputs": [],
   "source": [
    "def read_concorde_tour(filename):\n",
    "    with open(filename, 'r') as f:\n",
    "        n = None\n",
    "        tour = []\n",
    "        for line in f:\n",
    "            if n is None:\n",
    "                n = int(line)\n",
    "            else:\n",
    "                tour.extend([int(node) for node in line.rstrip().split(\" \")])\n",
    "    assert len(tour) == n, \"Unexpected tour length\"\n",
    "    return tour"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "5956ddfd-59c0-441b-b918-13bf07073eca",
   "metadata": {},
   "outputs": [],
   "source": [
    "def write_tsplib(filename, distance_matrix, name=\"problem\"):\n",
    "\n",
    "    with open(filename, 'w') as f:\n",
    "        f.write(\"\\n\".join([\n",
    "            \"{} : {}\".format(k, v)\n",
    "            for k, v in (\n",
    "                (\"NAME\", name),\n",
    "                (\"TYPE\", \"TSP\"),\n",
    "                (\"DIMENSION\", distance_matrix.shape[0]),\n",
    "                (\"EDGE_WEIGHT_TYPE\", \"EXPLICIT\"),\n",
    "                (\"EDGE_WEIGHT_FORMAT\", \"FULL_MATRIX\"),\n",
    "            )\n",
    "        ]))\n",
    "        f.write(\"\\n\")\n",
    "        f.write(\"EDGE_WEIGHT_SECTION\\n\")\n",
    "\n",
    "        for i in range(distance_matrix.shape[0]):\n",
    "            costs = [str(int(x*(10**7)+0.5)) for x in distance_matrix[i, :]]\n",
    "            #print(costs)\n",
    "            s = ' '.join(costs)\n",
    "\n",
    "            f.write(s)\n",
    "            f.write(\"\\n\")        \n",
    "        f.write(\"EOF\\n\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "85a521fe-7755-4244-9952-b4890996d051",
   "metadata": {},
   "outputs": [],
   "source": [
    "def solve_concorde_log(executable, directory, name, distance_matrix, disable_cache=False):\n",
    "\n",
    "    problem_filename = os.path.join(directory, \"{}.tsp\".format(name))\n",
    "    tour_filename = os.path.join(directory, \"{}.tour\".format(name))\n",
    "    output_filename = os.path.join(directory, \"{}.concorde.pkl\".format(name))\n",
    "    log_filename = os.path.join(directory, \"{}.log\".format(name))\n",
    "\n",
    "    # if True:\n",
    "    try:\n",
    "        # May have already been run\n",
    "        if os.path.isfile(output_filename) and not disable_cache:\n",
    "            tour, duration = load_dataset(output_filename)\n",
    "        else:\n",
    "            write_tsplib(problem_filename, distance_matrix, name=name)\n",
    "\n",
    "            with open(log_filename, 'w') as f:\n",
    "                start = time.time()\n",
    "                try:\n",
    "                    # Concorde is weird, will leave traces of solution in current directory so call from target dir\n",
    "                    check_call([executable, '-s', '1234', '-x', '-o',\n",
    "                                os.path.abspath(tour_filename), os.path.abspath(problem_filename)],\n",
    "                               stdout=f, stderr=f, cwd=directory)\n",
    "                except CalledProcessError as e:\n",
    "                    # Somehow Concorde returns 255\n",
    "                    assert e.returncode == 255\n",
    "                duration = time.time() - start\n",
    "\n",
    "            tour = read_concorde_tour(tour_filename)\n",
    "            #save_dataset((tour, duration), output_filename)\n",
    "\n",
    "        #return calc_tsp_length(loc, tour), tour, duration\n",
    "        return tour\n",
    "\n",
    "    except Exception as e:\n",
    "        print(\"Exception occured\")\n",
    "        print(e)\n",
    "        return None"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "141312dd-a23f-49d6-b05f-f30a586860fb",
   "metadata": {},
   "outputs": [],
   "source": [
    "TRIALS = 100\n",
    "N_CITIES = 300"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "f09f5c27-41ef-4334-b818-61d5c4f74cbf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CPU times: user 8.09 s, sys: 288 ms, total: 8.38 s\n",
      "Wall time: 3min 55s\n"
     ]
    }
   ],
   "source": [
    "%%time\n",
    "\n",
    "data_concorde = []\n",
    "\n",
    "for t in range(TRIALS):\n",
    "    distance_matrix = create_problem_nonmetric(t, N_CITIES)\n",
    "    best_found_route = solve_concorde_log('/concorde_src/concorde/TSP/concorde', '.', 'test', distance_matrix)\n",
    "    data_concorde.append(calculate_total_distance(best_found_route, distance_matrix))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "id": "006e109a-a0c7-4f74-ad1a-23f953ddab0e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(1.025097733909528, 0.005752742868348526)"
      ]
     },
     "execution_count": 40,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.mean(data_concorde), np.std(data_concorde) / np.sqrt(TRIALS-1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "087430d9-ba0d-4e48-9aa9-c135f5cfd272",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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.11.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
