{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "fa96a1d4",
   "metadata": {},
   "source": [
    "need to refresh everytime you change the target folder  by copying from network folder to combined folder\n",
    "to test or evaluate performance of llm generated code for stalled vehicle we need to chasnge target folder \n",
    "there are total 8 target folders starting from NE12_50\n",
    "Finally after getting same name csv files from all the folders we need to find average before and after llm logic"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1e4e65ee",
   "metadata": {},
   "source": [
    "copy code"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "814b3cf2",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "import os\n",
    "import shutil\n",
    "\n",
    "# ─────────── USER CONFIG ───────────\n",
    "BASE_DIR = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\"\n",
    "TARGET_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\\NE12_50\"\n",
    "\n",
    "#TARGET_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\\Bellevue_150th_Newport__2017-09-11_17-08-32\"\n",
    "\n",
    "#TARGET_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\\Bellevue_150th_Newport__2017-09-11_08-08-31\"\n",
    "\n",
    "##TARGET_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\\Bellevue_150th_Newport__2017-09-10_18-08-24\"\n",
    "NETWORK_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\new_road_network\"\n",
    "DURATION_FOLDERS = [\"2min\", \"3min\", \"4min\", \"5min\", \"6min\"]\n",
    "\n",
    "# ─────────── PROCESS EACH FOLDER ───────────\n",
    "for duration in DURATION_FOLDERS:\n",
    "    duration_path = os.path.join(TARGET_FOLDER, duration)\n",
    "    combined_dir = os.path.join(duration_path, \"combined\")\n",
    "    os.makedirs(combined_dir, exist_ok=True)\n",
    "\n",
    "    # Copy route.rou.xml files from each Video folder\n",
    "    counter = 1\n",
    "    for folder in sorted(os.listdir(duration_path)):\n",
    "        if folder.lower().startswith(\"video\"):\n",
    "            route_src = os.path.join(duration_path, folder, \"route.rou.xml\")\n",
    "            if os.path.exists(route_src):\n",
    "                route_dst = os.path.join(combined_dir, f\"route.rou{counter}.xml\")\n",
    "                shutil.copy(route_src, route_dst)\n",
    "                print(f\"✅ Copied {folder}/route.rou.xml → {duration}/combined/route.rou{counter}.xml\")\n",
    "                counter += 1\n",
    "\n",
    "    # Copy all network files except route.rou.xml\n",
    "    # Copy all network files except route.rou.xml\n",
    "    for file in os.listdir(NETWORK_FOLDER):\n",
    "      if (file.endswith(\".xml\") and not file.startswith(\"route.rou\")) or file == \"sumo_config.sumocfg\":\n",
    "        src = os.path.join(NETWORK_FOLDER, file)\n",
    "        dst = os.path.join(combined_dir, file)\n",
    "        shutil.copy(src, dst)\n",
    "        print(f\"📄 Copied {file} → {duration}/combined\")\n",
    "\n",
    "\n",
    "print(\"\\n🎯 All files copied successfully.\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a4794a1d",
   "metadata": {},
   "source": [
    "stalled vehicle near west left turn pocket"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ff037877",
   "metadata": {},
   "source": [
    "change target folder to test other days data  , always run the copy code first when testing different day data  \n",
    "this is before llm code . each time we change the target folder first run the copy code to copy files from from network to combined folder\n",
    "other target folders are : Bellevue_150th_Newport__2017-09-11_17-08-32 ,Bellevue_150th_Newport__2017-09-11_08-08-31 etc"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "26c00d5c",
   "metadata": {},
   "source": [
    "scenario : stalled vehicle near west left turn pocket"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a60abc0d",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import shutil\n",
    "import xml.etree.ElementTree as ET\n",
    "import traci\n",
    "import pandas as pd\n",
    "import time\n",
    "import gc\n",
    "from pathlib import Path\n",
    "from sumolib import checkBinary\n",
    "import psutil\n",
    "\n",
    "# ─────────── USER CONFIG ───────────\n",
    "BASE_DIR = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\"\n",
    "TARGET_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\\NE12_50\"\n",
    "#run the copy code then run this code  with new target folder \n",
    "#TARGET_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\\Bellevue_150th_Newport__2017-09-11_17-08-32\"\n",
    "#run the copy code then run this code  with new target folder \n",
    "#TARGET_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\\Bellevue_150th_Newport__2017-09-11_08-08-31\"\n",
    "#run the copy code then run this code  with new target folder \n",
    "##TARGET_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\\Bellevue_150th_Newport__2017-09-10_18-08-24\"\n",
    "#total 8 target folders\n",
    "SUMO_BINARY = r\"C:\\Program Files (x86)\\Eclipse\\Sumo\\bin\\sumo.exe\"\n",
    "STEP = 1\n",
    "CYCLE_SEC = 90\n",
    "NETWORK_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\new_road_network\"\n",
    "# ─────────── BROKEN CAR CONFIG ───────────\n",
    "\n",
    "\n",
    "BROKEN_ROUTE = \"route_west_to_north\"\n",
    "BROKEN_LANE = \"west_to_center_2\"\n",
    "\n",
    "#if the stalled car was in  other 3 directions north , south , east : \n",
    "\n",
    "#if the stalled car was near north left turn \n",
    "#BROKEN_ROUTE = \"route_north_to_east\"\n",
    "#BROKEN_LANE = \"north_to_center_2\"\n",
    "\n",
    "#if the stalled car was near south left turn \n",
    "#BROKEN_ROUTE = \"route_south_to_west\"\n",
    "#BROKEN_LANE = \"south_to_center_2\"\n",
    "\n",
    "#if the stalled car was near east left turn \n",
    "#BROKEN_ROUTE = \"route_east_to_south\"\n",
    "#BROKEN_LANE = \"east_to_center_2\"\n",
    "\n",
    "\n",
    "DEPART = 2.40\n",
    "STALL = 800\n",
    "END = 2300\n",
    "TOW_TIME = DEPART + STALL\n",
    "\n",
    "def kill_sumo():\n",
    "    for proc in psutil.process_iter(['pid', 'name']):\n",
    "        if 'sumo' in proc.info['name'].lower():\n",
    "            proc.kill()\n",
    "\n",
    "def move_routes_and_setup_combined(day_path, duration_folder):\n",
    "    combined_dir = os.path.join(day_path, duration_folder, \"combined\")\n",
    "\n",
    "    # Ensure all files are closed and garbage collected before deletion\n",
    "    time.sleep(1)\n",
    "    gc.collect()\n",
    "\n",
    "    # Optional: kill SUMO if still running\n",
    "    # kill_sumo()\n",
    "\n",
    "    if os.path.exists(combined_dir):\n",
    "        try:\n",
    "            shutil.rmtree(combined_dir)\n",
    "        except PermissionError as e:\n",
    "            print(f\"❌ Permission error while deleting {combined_dir}: {e}\")\n",
    "            print(\"⏳ Waiting 2 seconds before retrying...\")\n",
    "            time.sleep(2)\n",
    "            shutil.rmtree(combined_dir)\n",
    "    os.makedirs(combined_dir, exist_ok=True)\n",
    "\n",
    "    duration_path = os.path.join(day_path, duration_folder)\n",
    "    counter = 1\n",
    "    for folder in sorted(os.listdir(duration_path)):\n",
    "        if folder.lower().startswith(\"video\"):\n",
    "            route_src = os.path.join(duration_path, folder, \"route.rou.xml\")\n",
    "            if os.path.exists(route_src):\n",
    "                shutil.copy(route_src, os.path.join(combined_dir, f\"route.rou{counter}.xml\"))\n",
    "                counter += 1\n",
    "    if os.path.exists(NETWORK_FOLDER):\n",
    "        for file in os.listdir(NETWORK_FOLDER):\n",
    "            if file.endswith((\".net.xml\", \".nod.xml\", \".edg.xml\", \".typ.xml\", \".sumocfg\", \".add.xml\", \".xml\")):\n",
    "                shutil.copy(os.path.join(NETWORK_FOLDER, file), os.path.join(combined_dir, file))\n",
    "    return combined_dir\n",
    "\n",
    "def insert_broken_vehicle(xml_path):\n",
    "    tree = ET.parse(xml_path)\n",
    "    root = tree.getroot()\n",
    "\n",
    "    for v in root.findall('vehicle[@id=\"broken\"]'):\n",
    "        root.remove(v)\n",
    "\n",
    "    v = ET.SubElement(root, \"vehicle\",\n",
    "                      id=\"broken\",\n",
    "                      route=BROKEN_ROUTE,\n",
    "                      depart=str(DEPART),\n",
    "                      color=\"255,120,0\")\n",
    "    ET.SubElement(v, \"stop\", lane=BROKEN_LANE, pos=\"82\", duration=\"999999\")\n",
    "\n",
    "    others = [e for e in root if e.tag != \"vehicle\"]\n",
    "    vehicles = sorted(root.findall(\"vehicle\"), key=lambda x: float(x.get('depart', '0')))\n",
    "    root[:] = others + vehicles\n",
    "\n",
    "    tree.write(xml_path, encoding=\"utf-8\", xml_declaration=True)\n",
    "\n",
    "def update_sumo_config_route_file(cfg_file, new_route_file):\n",
    "    tree = ET.parse(cfg_file)\n",
    "    root = tree.getroot()\n",
    "\n",
    "    for input_elem in root.findall(\"input\"):\n",
    "        for child in input_elem.findall(\"route-files\"):\n",
    "            child.set(\"value\", new_route_file)\n",
    "\n",
    "        for old in input_elem.findall(\"additional-files\"):\n",
    "            input_elem.remove(old)\n",
    "\n",
    "    tree.write(cfg_file, encoding=\"utf-8\", xml_declaration=True)\n",
    "\n",
    "def run_simulation(cfg_file, combined_folder):\n",
    "    results = []\n",
    "\n",
    "    for i in range(1, 50):\n",
    "        route_file = f\"route.rou{i}.xml\"\n",
    "        route_path = os.path.join(combined_folder, route_file)\n",
    "        if not os.path.exists(route_path):\n",
    "            continue\n",
    "\n",
    "        insert_broken_vehicle(route_path)\n",
    "        update_sumo_config_route_file(cfg_file, route_file)\n",
    "\n",
    "        traci.start([\n",
    "            SUMO_BINARY, \"-c\", cfg_file,\n",
    "           \n",
    "            \"--begin\", \"0\", \"--end\", str(END),\n",
    "            \"--collision.action\", \"none\",\n",
    "            \"--time-to-teleport\", \"-1\"\n",
    "        ])\n",
    "\n",
    "        depart_times = {}\n",
    "        arrival_times = {}\n",
    "        waiting_times = {}\n",
    "\n",
    "        next_cut = CYCLE_SEC\n",
    "\n",
    "        while traci.simulation.getMinExpectedNumber() > 0:\n",
    "            traci.simulationStep()\n",
    "            t = traci.simulation.getTime()\n",
    "\n",
    "            if t >= TOW_TIME and \"broken\" in traci.vehicle.getIDList():\n",
    "                traci.vehicle.remove(\"broken\")\n",
    "\n",
    "            for veh_id in traci.vehicle.getIDList():\n",
    "                if veh_id not in depart_times:\n",
    "                    depart_times[veh_id] = t\n",
    "                waiting_times[veh_id] = traci.vehicle.getAccumulatedWaitingTime(veh_id)\n",
    "\n",
    "            if t >= next_cut:\n",
    "                next_cut += CYCLE_SEC\n",
    "\n",
    "        for veh_id in traci.simulation.getArrivedIDList():\n",
    "            arrival_times[veh_id] = traci.simulation.getTime()\n",
    "\n",
    "        traci.close()\n",
    "        time.sleep(1)  # Ensure file handles are released\n",
    "\n",
    "        travel_times = []\n",
    "        wait_times = []\n",
    "\n",
    "        for veh_id in arrival_times:\n",
    "            if veh_id in depart_times:\n",
    "                travel_time = arrival_times[veh_id] - depart_times[veh_id]\n",
    "                wait_time = waiting_times.get(veh_id, 0)\n",
    "                travel_times.append(travel_time)\n",
    "                wait_times.append(wait_time)\n",
    "\n",
    "        ATT = sum(travel_times) / len(travel_times) if travel_times else 0\n",
    "        AWT = sum(wait_times) / len(wait_times) if wait_times else 0\n",
    "\n",
    "        print(f\"\\n✅ Simulation {route_file} ended\")\n",
    "        print(f\"⏱️ ATT: {ATT:.2f} s | AWT: {AWT:.2f} s\")\n",
    "\n",
    "        results.append({\n",
    "            \"route_file\": route_file,\n",
    "            \"ATT\": float(f\"{ATT:.2f}\"),\n",
    "            \"AWT\": float(f\"{AWT:.2f}\")\n",
    "        })\n",
    "\n",
    "    return results\n",
    "\n",
    "# ─────────── OUTER LOOP OVER VIDEO FOLDERS ───────────\n",
    "day_path = os.path.join(BASE_DIR, TARGET_FOLDER)\n",
    "all_results = []\n",
    "\n",
    "for subfolder in os.listdir(day_path):\n",
    "    if subfolder.lower() in [\"2min\", \"3min\", \"4min\", \"5min\", \"6min\"]:\n",
    "        combined_dir = move_routes_and_setup_combined(day_path, subfolder)\n",
    "        cfg_file_path = os.path.join(combined_dir, \"sumo_config.sumocfg\")\n",
    "\n",
    "        results = run_simulation(cfg_file_path, combined_dir)\n",
    "        for r in results:\n",
    "            r[\"duration_folder\"] = subfolder\n",
    "        all_results.extend(results)\n",
    "\n",
    "csv_path = os.path.join(TARGET_FOLDER, \"broken_car_metrics_in_west.csv\") #record or store each day's metrics in csv file \n",
    "results_df = pd.DataFrame(all_results)\n",
    "results_df.to_csv(csv_path, index=False, float_format=\"%.2f\")\n",
    "\n",
    "print(f\"\\n✅ Final results saved: {csv_path}\")\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e5cf3aa7",
   "metadata": {},
   "source": [
    "refresh each time,run the copy code each time  and change the target folder , record or store each day's metrics in csv file "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "accb927c",
   "metadata": {},
   "source": [
    "after llm generated logic \n",
    "here target folder : NE12_50\n",
    "but other day's folders also need to be tested as well : Bellevue_150th_Newport__2017-09-11_17-08-32 ,Bellevue_150th_Newport__2017-09-11_08-08-31 etc"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "915e7e0c",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import shutil\n",
    "import xml.etree.ElementTree as ET\n",
    "import traci\n",
    "import pandas as pd\n",
    "from sumolib import checkBinary\n",
    "\n",
    "# ─────────── USER CONFIG ───────────\n",
    "BASE_DIR = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\"\n",
    "TARGET_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\\NE12_50\"\n",
    "#run the copy code then run this code  with new target folder \n",
    "#TARGET_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\\Bellevue_150th_Newport__2017-09-11_17-08-32\"\n",
    "#run the copy code then run this code  with new target folder \n",
    "#TARGET_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\\Bellevue_150th_Newport__2017-09-11_08-08-31\"\n",
    "#run the copy code then run this code  with new target folder \n",
    "##TARGET_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\\Bellevue_150th_Newport__2017-09-10_18-08-24\"\n",
    "SUMO_BINARY = r\"C:\\Program Files (x86)\\Eclipse\\Sumo\\bin\\sumo.exe\"\n",
    "STEP = 1\n",
    "CYCLE_SEC = 90\n",
    "NETWORK_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\new_road_network\"\n",
    "\n",
    "# ─────────── BROKEN CAR + SWITCH CONFIG ───────────\n",
    "\n",
    "BROKEN_ROUTE = \"route_west_to_north\"\n",
    "BROKEN_LANE = \"west_to_center_2\"\n",
    "\n",
    "#for other 3 directions north , south , east\n",
    "#if stalled car was near north left turn \n",
    "#BROKEN_ROUTE = \"route_north_to_east\"\n",
    "#BROKEN_LANE = \"north_to_center_2\"\n",
    "\n",
    "#if stalled car was near south left turn \n",
    "#BROKEN_ROUTE = \"route_south_to_west\"\n",
    "#BROKEN_LANE = \"south_to_center_2\"\n",
    "\n",
    "#if stalled car was near east left turn \n",
    "#BROKEN_ROUTE = \"route_east_to_south\"\n",
    "#BROKEN_LANE = \"east_to_center_2\"\n",
    "\n",
    "DEPART = 2.40\n",
    "STALL = 800\n",
    "END = 2300\n",
    "TOW_TIME = DEPART + STALL\n",
    "TLS_SWITCH_PROGRAM = \"faulty_EL\"\n",
    "TLS_ID = \"center\"\n",
    "BROKEN_VEHICLE_ID = \"broken\"\n",
    "\n",
    "\n",
    "\n",
    "def move_routes_and_setup_combined(day_path, duration_folder):\n",
    "    combined_dir = os.path.join(day_path, duration_folder, \"combined\")\n",
    "    if os.path.exists(combined_dir):\n",
    "        shutil.rmtree(combined_dir)\n",
    "    os.makedirs(combined_dir, exist_ok=True)\n",
    "\n",
    "    duration_path = os.path.join(day_path, duration_folder)\n",
    "    counter = 1\n",
    "    for folder in sorted(os.listdir(duration_path)):\n",
    "        if folder.lower().startswith(\"video\"):\n",
    "            route_src = os.path.join(duration_path, folder, \"route.rou.xml\")\n",
    "            if os.path.exists(route_src):\n",
    "                shutil.copy(route_src, os.path.join(combined_dir, f\"route.rou{counter}.xml\"))\n",
    "                counter += 1\n",
    "\n",
    "    if os.path.exists(NETWORK_FOLDER):\n",
    "        for file in os.listdir(NETWORK_FOLDER):\n",
    "            src = os.path.join(NETWORK_FOLDER, file)\n",
    "            dst = os.path.join(combined_dir, file)\n",
    "            if file.endswith((\".net.xml\", \".nod.xml\", \".edg.xml\", \".typ.xml\", \".sumocfg\", \".add.xml\", \".xml\")):\n",
    "                shutil.copy(src, dst)\n",
    "\n",
    "    return combined_dir\n",
    "\n",
    "\n",
    "def insert_broken_vehicle(xml_path):\n",
    "    tree = ET.parse(xml_path)\n",
    "    root = tree.getroot()\n",
    "\n",
    "    for v in root.findall(f'vehicle[@id=\"{BROKEN_VEHICLE_ID}\"]'):\n",
    "        root.remove(v)\n",
    "\n",
    "    v = ET.SubElement(root, \"vehicle\",\n",
    "                      id=BROKEN_VEHICLE_ID,\n",
    "                      route=BROKEN_ROUTE,\n",
    "                      depart=str(DEPART),\n",
    "                      color=\"255,120,0\")\n",
    "    ET.SubElement(v, \"stop\", lane=BROKEN_LANE, pos=\"82\", duration=\"999999\")\n",
    "\n",
    "    others = [e for e in root if e.tag != \"vehicle\"]\n",
    "    vehicles = sorted(root.findall(\"vehicle\"), key=lambda x: float(x.get('depart', '0')))\n",
    "    root[:] = others + vehicles\n",
    "\n",
    "    tree.write(xml_path, encoding=\"utf-8\", xml_declaration=True)\n",
    "\n",
    "\n",
    "def update_sumo_config_route_file(cfg_file, new_route_file):\n",
    "    tree = ET.parse(cfg_file)\n",
    "    root = tree.getroot()\n",
    "\n",
    "    for input_elem in root.findall(\"input\"):\n",
    "        for child in input_elem.findall(\"route-files\"):\n",
    "            child.set(\"value\", new_route_file)\n",
    "\n",
    "        # Remove old additional-files\n",
    "        for old in input_elem.findall(\"additional-files\"):\n",
    "            input_elem.remove(old)\n",
    "\n",
    "        # Add traffic light logic file\n",
    "        ET.SubElement(input_elem, \"additional-files\", {\"value\": \"generated_broken_wleft.add.xml\"})\n",
    "\n",
    "    tree.write(cfg_file, encoding=\"utf-8\", xml_declaration=True)\n",
    "\n",
    "\n",
    "def run_simulation(cfg_file, combined_folder):\n",
    "    results = []\n",
    "\n",
    "    for i in range(1, 50):\n",
    "        route_file = f\"route.rou{i}.xml\"\n",
    "        route_path = os.path.join(combined_folder, route_file)\n",
    "        if not os.path.exists(route_path):\n",
    "            continue\n",
    "\n",
    "        insert_broken_vehicle(route_path)\n",
    "        update_sumo_config_route_file(cfg_file, route_file)\n",
    "\n",
    "        traci.start([\n",
    "            SUMO_BINARY, \"-c\", cfg_file,\n",
    "           \n",
    "            \"--begin\", \"0\", \"--end\", str(END),\n",
    "            \"--collision.action\", \"none\",\n",
    "            \"--time-to-teleport\", \"-1\",\n",
    "            \"--no-warnings\"\n",
    "        ])\n",
    "\n",
    "        switch_done = False\n",
    "       \n",
    "        depart_times, arrival_times, waiting_times = {}, {}, {}\n",
    "        next_cut = CYCLE_SEC\n",
    "\n",
    "        while True:\n",
    "            traci.simulationStep()\n",
    "            sim_time = traci.simulation.getTime()\n",
    "\n",
    "            # Vehicle towing and signal switch\n",
    "            if sim_time >= TOW_TIME and not switch_done:\n",
    "                if BROKEN_VEHICLE_ID in traci.vehicle.getIDList():\n",
    "                    print(f\"🚓 {sim_time:.2f}s: Towing '{BROKEN_VEHICLE_ID}'\")\n",
    "                    traci.vehicle.remove(BROKEN_VEHICLE_ID)\n",
    "\n",
    "                print(f\"⏱️ {sim_time:.2f}s: Switching traffic signal to '{TLS_SWITCH_PROGRAM}'\")\n",
    "                traci.trafficlight.setProgram(TLS_ID, TLS_SWITCH_PROGRAM)\n",
    "                switch_done = True\n",
    "\n",
    "            \n",
    "\n",
    "            for vid in traci.vehicle.getIDList():\n",
    "                if vid not in depart_times:\n",
    "                    depart_times[vid] = sim_time\n",
    "                waiting_times[vid] = traci.vehicle.getAccumulatedWaitingTime(vid)\n",
    "\n",
    "            if sim_time >= next_cut:\n",
    "                next_cut += CYCLE_SEC\n",
    "\n",
    "            if sim_time >= END or traci.simulation.getMinExpectedNumber() == 0:\n",
    "                print(f\"✅ Simulation ended at {sim_time:.2f}s\")\n",
    "                break\n",
    "\n",
    "        for vid in traci.simulation.getArrivedIDList():\n",
    "            arrival_times[vid] = traci.simulation.getTime()\n",
    "\n",
    "        traci.close()\n",
    "\n",
    "        travel_times = []\n",
    "        wait_times = []\n",
    "\n",
    "        for vid in arrival_times:\n",
    "            if vid in depart_times:\n",
    "                tt = arrival_times[vid] - depart_times[vid]\n",
    "                wt = waiting_times.get(vid, 0)\n",
    "                travel_times.append(tt)\n",
    "                wait_times.append(wt)\n",
    "\n",
    "        ATT = sum(travel_times) / len(travel_times) if travel_times else 0\n",
    "        AWT = sum(wait_times) / len(wait_times) if wait_times else 0\n",
    "\n",
    "        \n",
    "\n",
    "        print(f\"\\n✅ Simulation {route_file} results:\")\n",
    "        \n",
    "        print(f\"⏱️ ATT: {ATT:.2f}s | AWT: {AWT:.2f}s\")\n",
    "\n",
    "        results.append({\n",
    "            \"route_file\": route_file,\n",
    "            \"ATT\": round(ATT, 2),\n",
    "            \"AWT\": round(AWT, 2)\n",
    "        })\n",
    "\n",
    "    return results\n",
    "\n",
    "\n",
    "# ─────────── RUN ALL ───────────\n",
    "day_path = os.path.join(BASE_DIR, TARGET_FOLDER)\n",
    "all_results = []\n",
    "\n",
    "for subfolder in os.listdir(day_path):\n",
    "    if subfolder.lower() in [\"2min\", \"3min\", \"4min\", \"5min\", \"6min\"]:\n",
    "        combined_dir = move_routes_and_setup_combined(day_path, subfolder)\n",
    "        cfg_file_path = os.path.join(combined_dir, \"sumo_config.sumocfg\")\n",
    "\n",
    "        results = run_simulation(cfg_file_path, combined_dir)\n",
    "        for r in results:\n",
    "            r[\"duration_folder\"] = subfolder\n",
    "        all_results.extend(results)\n",
    "\n",
    "csv_path = os.path.join(TARGET_FOLDER, \"broken_car_metrics_in_west_logic.csv\")  #record or store each day's metrics in csv file \n",
    "pd.DataFrame(all_results).to_csv(csv_path, index=False, float_format=\"%.2f\")\n",
    "\n",
    "print(f\"\\n📁 Saved: {csv_path}\")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f6b36443",
   "metadata": {},
   "source": [
    "refresh each time,run the copy code each time  and change the target folder , record or store each day's metrics in csv file "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5813fe95",
   "metadata": {},
   "source": [
    "overall average of all days data before llm logic : overall average of all 8 target folders/day's data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5c7bbe30",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "✅ Average ATT: 932.88\n",
      "✅ Average AWT: 63.59\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import pandas as pd\n",
    "\n",
    "# Base directory path\n",
    "base_dir = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\"\n",
    "csv_filename = \"broken_car_metrics_in_west.csv\"  # same file name in all subfolders\n",
    "# if north csv_filename = \"broken_car_metrics_in_north.csv\" \n",
    "# if south csv_filename = \"broken_car_metrics_in_south.csv\" \n",
    "# if east csv_filename = \"broken_car_metrics_in_east.csv\" \n",
    "\n",
    "\n",
    "att_values = []\n",
    "awt_values = []\n",
    "\n",
    "# Loop through all subdirectories\n",
    "for folder_name in os.listdir(base_dir):\n",
    "    \n",
    "    folder_path = os.path.join(base_dir, folder_name)\n",
    "    if os.path.isdir(folder_path):\n",
    "        file_path = os.path.join(folder_path, csv_filename)\n",
    "        if os.path.exists(file_path):\n",
    "            try:\n",
    "                df = pd.read_csv(file_path)\n",
    "                if 'ATT' in df.columns and 'AWT' in df.columns:\n",
    "                    \n",
    "                    att_values.extend(df['ATT'].dropna().tolist())\n",
    "                    awt_values.extend(df['AWT'].dropna().tolist())\n",
    "            except Exception as e:\n",
    "                print(f\"Error reading {file_path}: {e}\")\n",
    "\n",
    "# Calculate averages\n",
    "\n",
    "if att_values and awt_values:\n",
    "    avg_att = sum(att_values) / len(att_values)\n",
    "    avg_awt = sum(awt_values) / len(awt_values)\n",
    "    print(f\"✅ Average ATT: {avg_att:.2f}\")\n",
    "    print(f\"✅ Average AWT: {avg_awt:.2f}\")\n",
    "else:\n",
    "    print(\"⚠️ No valid ATT and AWT values found in the subfolders.\")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "38bfa241",
   "metadata": {},
   "source": [
    "overall average of all days data after llm logic : overall average of all 8 target folders/day's data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bd95a2f1",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "✅ Average ATT: 721.27\n",
      "✅ Average AWT: 48.08\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import pandas as pd\n",
    "\n",
    "# Base directory path\n",
    "base_dir = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\"\n",
    "csv_filename = \"broken_car_metrics_in_west_logic.csv\"  # same file name in all subfolders\n",
    "# if north csv_filename = \"broken_car_metrics_in_north_logic.csv\" \n",
    "# if south csv_filename = \"broken_car_metrics_in_south_logic.csv\" \n",
    "# if east csv_filename = \"broken_car_metrics_in_east_logic.csv\" \n",
    "\n",
    "att_values = []\n",
    "awt_values = []\n",
    "\n",
    "# Loop through all subdirectories\n",
    "for folder_name in os.listdir(base_dir):\n",
    "    folder_path = os.path.join(base_dir, folder_name)\n",
    "    if os.path.isdir(folder_path):\n",
    "        file_path = os.path.join(folder_path, csv_filename)\n",
    "        if os.path.exists(file_path):\n",
    "            try:\n",
    "                df = pd.read_csv(file_path)\n",
    "                if 'ATT' in df.columns and 'AWT' in df.columns:\n",
    "                    att_values.extend(df['ATT'].dropna().tolist())\n",
    "                    awt_values.extend(df['AWT'].dropna().tolist())\n",
    "            except Exception as e:\n",
    "                print(f\"Error reading {file_path}: {e}\")\n",
    "\n",
    "# Calculate averages\n",
    "if att_values and awt_values:\n",
    "    avg_att = sum(att_values) / len(att_values)\n",
    "    avg_awt = sum(awt_values) / len(awt_values)\n",
    "    print(f\"✅ Average ATT: {avg_att:.2f}\")\n",
    "    print(f\"✅ Average AWT: {avg_awt:.2f}\")\n",
    "else:\n",
    "    print(\"⚠️ No valid ATT and AWT values found in the subfolders.\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a59b5b3d",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.13.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
