{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "06cbfe70",
   "metadata": {},
   "source": []
  },
  {
   "cell_type": "markdown",
   "id": "745bc00f",
   "metadata": {},
   "source": [
    "refresh and copy code  everytime you test new scenario"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "e397d62b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "✅ All files copied successfully.\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import shutil\n",
    "\n",
    "# ───── Paths ─────\n",
    "network_src = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\new_road_network\"\n",
    "route_src = 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\\2Min\\Video_1\\route.rou.xml\"\n",
    "dest_folder = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\testing_3_scenarios\"\n",
    "\n",
    "# ───── Create Destination Folder if Needed ─────\n",
    "os.makedirs(dest_folder, exist_ok=True)\n",
    "\n",
    "# ───── Copy All Files from new_road_network ─────\n",
    "for file in os.listdir(network_src):\n",
    "    src_path = os.path.join(network_src, file)\n",
    "    dst_path = os.path.join(dest_folder, file)\n",
    "    if os.path.isfile(src_path):\n",
    "        shutil.copy(src_path, dst_path)\n",
    "\n",
    "# ───── Copy route.rou.xml ─────\n",
    "shutil.copy(route_src, os.path.join(dest_folder, \"route.rou.xml\"))\n",
    "\n",
    "print(\"✅ All files copied successfully.\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0e8515a0",
   "metadata": {},
   "source": [
    "Stalled vehicle near left turn intersection"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "07a95e56",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "🚓 802.0s: Towing 'broken'\n",
      "✅ SUMO simulation complete.\n"
     ]
    }
   ],
   "source": [
    "import xml.etree.ElementTree as ET\n",
    "import traci\n",
    "from pathlib import Path\n",
    "from sumolib import checkBinary\n",
    "\n",
    "# ───── Configuration ─────\n",
    "CFG_FILE = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\testing_3_scenarios\\sumo_config.sumocfg\"\n",
    "ROUTE_FILE = Path(r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\testing_3_scenarios\\route.rou.xml\")\n",
    "BROKEN_ROUTE = \"route_west_to_north\"\n",
    "BROKEN_LANE = \"west_to_center_2\"\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",
    "DEPART = 1.32\n",
    "STALL = 800\n",
    "END = 2300\n",
    "\n",
    "# ───── Update route file ─────\n",
    "root = ET.parse(ROUTE_FILE).getroot()\n",
    "\n",
    "# Remove any existing broken vehicle\n",
    "for n in root.findall('vehicle[@id=\"broken\"]'):\n",
    "    root.remove(n)\n",
    "\n",
    "# Add new broken vehicle with long stop\n",
    "v = ET.SubElement(root, \"vehicle\",\n",
    "                  id=\"broken\", route=BROKEN_ROUTE,\n",
    "                  depart=str(DEPART), color=\"255,120,0\")\n",
    "ET.SubElement(v, \"stop\", lane=BROKEN_LANE, pos=\"82\", duration=\"999999\")\n",
    "\n",
    "# Sort vehicles by depart time\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",
    "ET.ElementTree(root).write(ROUTE_FILE, encoding=\"utf-8\", xml_declaration=True)\n",
    "\n",
    "# ───── Start SUMO Simulation ─────\n",
    "cmd = [checkBinary(\"sumo\"), \"-c\", CFG_FILE,\n",
    "       \"--begin\", \"0\", \"--end\", str(END),\n",
    "       \"--collision.action\", \"none\",\n",
    "       \"--time-to-teleport\", \"-1\"]\n",
    "\n",
    "traci.start(cmd)\n",
    "\n",
    "tow_time = DEPART + STALL\n",
    "\n",
    "# ───── Track Stats ─────\n",
    "travel_times = {}\n",
    "waiting_times = {}\n",
    "\n",
    "while True:\n",
    "    traci.simulationStep()\n",
    "    t = traci.simulation.getTime()\n",
    "\n",
    "    # Tow the broken vehicle\n",
    "    if t >= tow_time and \"broken\" in traci.vehicle.getIDList():\n",
    "        print(f\"🚓 {t:.1f}s: Towing 'broken'\")\n",
    "        traci.vehicle.remove(\"broken\")\n",
    "\n",
    "    # Record depart time\n",
    "    for vid in traci.simulation.getDepartedIDList():\n",
    "        travel_times[vid] = {\"depart\": t, \"wait\": 0}\n",
    "\n",
    "    # Record arrival time\n",
    "    for vid in traci.simulation.getArrivedIDList():\n",
    "        if vid in travel_times:\n",
    "            travel_times[vid][\"arrive\"] = t\n",
    "\n",
    "    # Accumulate waiting time\n",
    "    for vid in traci.vehicle.getIDList():\n",
    "        speed = traci.vehicle.getSpeed(vid)\n",
    "        if speed < 0.1:  # considered waiting\n",
    "            if vid in travel_times:\n",
    "                travel_times[vid][\"wait\"] += 1\n",
    "\n",
    "    if t >= END:\n",
    "        break\n",
    "\n",
    "traci.close()\n",
    "print(\"✅ SUMO simulation complete.\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9239affa",
   "metadata": {},
   "source": [
    "choose of two faulty tllogic "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2f62a0be",
   "metadata": {},
   "source": [
    "copy code for faulty traffic lights for starving north south approach"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9cbae58e",
   "metadata": {},
   "outputs": [],
   "source": [
    "import re\n",
    "\n",
    "# Path to your .net.xml file\n",
    "net_file_path = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\new_road_network_for_malfunction\\simple_nw_se.net.xml\"\n",
    "\n",
    "# Your faulty traffic signal logic for starving northsouth approach\n",
    "faulty_tl_logic = \"\"\"\n",
    "<tlLogic id=\"center\" type=\"static\" programID=\"faulty_traffic\" offset=\"0\">\n",
    "    <phase duration=\"200\" state=\"rrrrrGGGGGrrrrrGGGGG\"/>   \n",
    "    <phase duration=\"10\" state=\"rrrrryyyyyrrrrryyyyy\"/>\n",
    "    <phase duration=\"4\"  state=\"GGGGGrrrrrGGGGGrrrrr\"/>\n",
    "    <phase duration=\"10\" state=\"yyyyyrrrrryyyyyrrrrr\"/>\n",
    "</tlLogic>\n",
    "\"\"\"\n",
    "\n",
    "# Read the original file\n",
    "with open(net_file_path, \"r\", encoding=\"utf-8\") as f:\n",
    "    content = f.read()\n",
    "\n",
    "# Replace the existing <tlLogic id=\"center\"...> block\n",
    "updated_content = re.sub(\n",
    "    r\"<tlLogic id=\\\"center\\\".*?</tlLogic>\",\n",
    "    faulty_tl_logic.strip(),\n",
    "    content,\n",
    "    flags=re.DOTALL\n",
    ")\n",
    "\n",
    "# Write back to the file\n",
    "with open(net_file_path, \"w\", encoding=\"utf-8\") as f:\n",
    "    f.write(updated_content)\n",
    "\n",
    "print(\"✅ Replaced <tlLogic> block with faulty traffic signal logic.\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "043d6d6f",
   "metadata": {},
   "source": [
    "faulty traffic logic for starving east west approach"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0eb4c864",
   "metadata": {},
   "outputs": [],
   "source": [
    "import re\n",
    "\n",
    "# Path to your .net.xml file\n",
    "net_file_path = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\new_road_network_for_malfunction\\simple_nw_se.net.xml\"\n",
    "\n",
    "# Your faulty traffic signal logic for starving northsouth approach\n",
    "faulty_tl_logic = \"\"\"\n",
    "<tlLogic id=\"center\" type=\"static\" programID=\"faulty_traffic\" offset=\"0\">\n",
    "    <phase duration=\"4\" state=\"rrrrrGGGGGrrrrrGGGGG\"/>   \n",
    "    <phase duration=\"10\" state=\"rrrrryyyyyrrrrryyyyy\"/>\n",
    "    <phase duration=\"200\"  state=\"GGGGGrrrrrGGGGGrrrrr\"/>\n",
    "    <phase duration=\"10\" state=\"yyyyyrrrrryyyyyrrrrr\"/>\n",
    "</tlLogic>\n",
    "\"\"\"\n",
    "\n",
    "# Read the original file\n",
    "with open(net_file_path, \"r\", encoding=\"utf-8\") as f:\n",
    "    content = f.read()\n",
    "\n",
    "# Replace the existing <tlLogic id=\"center\"...> block\n",
    "updated_content = re.sub(\n",
    "    r\"<tlLogic id=\\\"center\\\".*?</tlLogic>\",\n",
    "    faulty_tl_logic.strip(),\n",
    "    content,\n",
    "    flags=re.DOTALL\n",
    ")\n",
    "\n",
    "# Write back to the file\n",
    "with open(net_file_path, \"w\", encoding=\"utf-8\") as f:\n",
    "    f.write(updated_content)\n",
    "\n",
    "print(\"✅ Replaced <tlLogic> block with faulty traffic signal logic.\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "id": "86967e22",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "✅ All files copied successfully.\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import shutil\n",
    "\n",
    "# ───── Paths ─────\n",
    "network_src = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\new_road_network_for_malfunction\"\n",
    "route_src = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Bellevue_116th_NE12th__2017-09-11_09-08-31\\2Min\\Video_1\\route.rou.xml\"\n",
    "dest_folder = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\testing_3_scenarios\"\n",
    "\n",
    "# ───── Create Destination Folder if Needed ─────\n",
    "os.makedirs(dest_folder, exist_ok=True)\n",
    "\n",
    "# ───── Copy All Files from new_road_network ─────\n",
    "for file in os.listdir(network_src):\n",
    "    src_path = os.path.join(network_src, file)\n",
    "    dst_path = os.path.join(dest_folder, file)\n",
    "    if os.path.isfile(src_path):\n",
    "        shutil.copy(src_path, dst_path)\n",
    "\n",
    "# ───── Copy route.rou.xml ─────\n",
    "shutil.copy(route_src, os.path.join(dest_folder, \"route.rou.xml\"))\n",
    "\n",
    "print(\"✅ All files copied successfully.\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "204f3a2d",
   "metadata": {},
   "source": [
    "copy code for sudden burst "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "b499af91",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "✅ All files copied successfully.\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import shutil\n",
    "\n",
    "# ───── Paths ─────\n",
    "network_src = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\new_road_network\"\n",
    "route_src = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Bellevue_116th_NE12th__2017-09-11_09-08-31\\2Min\\Video_1\\route.rou.xml\"\n",
    "dest_folder = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\testing_3_scenarios\"\n",
    "\n",
    "# ───── Create Destination Folder if Needed ─────\n",
    "os.makedirs(dest_folder, exist_ok=True)\n",
    "\n",
    "# ───── Copy All Files from new_road_network ─────\n",
    "for file in os.listdir(network_src):\n",
    "    src_path = os.path.join(network_src, file)\n",
    "    dst_path = os.path.join(dest_folder, file)\n",
    "    if os.path.isfile(src_path):\n",
    "        shutil.copy(src_path, dst_path)\n",
    "\n",
    "# ───── Copy route.rou.xml ─────\n",
    "shutil.copy(route_src, os.path.join(dest_folder, \"route.rou.xml\"))\n",
    "\n",
    "print(\"✅ All files copied successfully.\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c0ac6a3a",
   "metadata": {},
   "source": [
    "sudden burst of vehicles "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e7e76054",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "✅ route.rou.xml copied.\n",
      "🚗 Sudden burst added for route_east_to_north.\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import xml.etree.ElementTree as ET\n",
    "import shutil\n",
    "\n",
    "# ───── CONFIG ─────\n",
    "network_src = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\new_road_network\"\n",
    "route_src = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Bellevue_116th_NE12th__2017-09-11_09-08-31\\2Min\\Video_1\\route.rou.xml\"\n",
    "dest_folder = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\testing_3_scenarios\"\n",
    "burst_route = \"route_east_to_north\"  # You can change this to other routes total 12 routes \n",
    "#  \"route_north_to_south\",\n",
    "  #   \"route_north_to_east\",\n",
    "   #  \"route_north_to_west\",\n",
    "    \n",
    "   #  \"route_east_to_south\",\n",
    "   #  \"route_east_to_west\",\n",
    "   #  \"route_east_to_north\",\n",
    "    \n",
    "   #  \"route_south_to_north\",\n",
    "   #  \"route_south_to_east\",\n",
    "   #  \"route_south_to_west\",\n",
    "    \n",
    "   #  \"route_west_to_north\",\n",
    "   #  \"route_west_to_east\",\n",
    "   #  \"route_west_to_south\"\n",
    "\n",
    "\n",
    "# ───── Copy route.rou.xml to destination ─────\n",
    "route_dst = os.path.join(dest_folder, \"route.rou.xml\")\n",
    "shutil.copy(route_src, route_dst)\n",
    "print(\"✅ route.rou.xml copied.\")\n",
    "\n",
    "# ───── Insert Burst Flow ─────\n",
    "def insert_burst_flow(xml_path, burst_route):\n",
    "    tree = ET.parse(xml_path)\n",
    "    root = tree.getroot()\n",
    "\n",
    "    # Add burst <flow>\n",
    "    burst_flow = ET.Element(\"flow\", {\n",
    "        \"id\": f\"burst_{burst_route}\",\n",
    "        \"begin\": \"10\",\n",
    "        \"end\": \"300\",\n",
    "        \"number\": \"270\",\n",
    "        \"route\": burst_route,\n",
    "        \"departLane\": \"best\",\n",
    "        \"departPos\": \"free\",\n",
    "        \"speedFactor\": \"1.0\",\n",
    "        \"speedDev\": \"0.0\"\n",
    "    })\n",
    "\n",
    "    # Sort vehicle insertions\n",
    "    others, vehicles_before, vehicles_after = [], [], []\n",
    "    for elem in root:\n",
    "        if elem.tag == \"vehicle\":\n",
    "            depart = float(elem.attrib.get(\"depart\", \"0\"))\n",
    "            (vehicles_before if depart < 10 else vehicles_after).append(elem)\n",
    "        else:\n",
    "            others.append(elem)\n",
    "\n",
    "    root.clear()\n",
    "    for e in others + vehicles_before + [burst_flow] + vehicles_after:\n",
    "        root.append(e)\n",
    "\n",
    "    tree.write(xml_path, encoding=\"utf-8\", xml_declaration=True)\n",
    "    print(f\"🚗 Sudden burst added for {burst_route}.\")\n",
    "\n",
    "# Apply burst insertion\n",
    "insert_burst_flow(route_dst, burst_route)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "adbea9bc",
   "metadata": {},
   "source": [
    "check detect which scenario is this stalled vehicle , or signal malfunction or suddden burst and print accordingly and store scenario in a flag variable "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "id": "adfc723b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " Traffic jam in NT+ST due to signal failure, estimated congestion: 29%\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import traci\n",
    "import xml.etree.ElementTree as ET\n",
    "from sumolib import checkBinary\n",
    "\n",
    "# ───── CONFIGURATION ─────\n",
    "SUMO_BINARY = checkBinary(\"sumo\")\n",
    "CFG_FILE = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\testing_3_scenarios\\sumo_config.sumocfg\"\n",
    "ROUTE_FILE = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\testing_3_scenarios\\route.rou.xml\"\n",
    "STEP = 1\n",
    "CYCLE_SEC = 90\n",
    "END_TIME = 2300\n",
    "scenario_flag = 0  # 1=stalled, 2=malfunction, 3=burst\n",
    "\n",
    "lane_groups = {\n",
    "    'NT': ['north_to_center_0', 'north_to_center_1'],\n",
    "    'ST': ['south_to_center_0', 'south_to_center_1'],\n",
    "    'N-L': ['north_to_center_2'],\n",
    "    'S-L': ['south_to_center_2'],\n",
    "    'ET': ['east_to_center_0', 'east_to_center_1'],\n",
    "    'WT': ['west_to_center_0', 'west_to_center_1'],\n",
    "    'E-L': ['east_to_center_2'],\n",
    "    'W-L': ['west_to_center_2']\n",
    "}\n",
    "\n",
    "phase_groups = {\n",
    "    'NT+ST': ['NT', 'ST'],\n",
    "    'NS-LEFTS': ['N-L', 'S-L'],\n",
    "    'ET+WT': ['ET', 'WT'],\n",
    "    'EW-LEFTS': ['E-L', 'W-L']\n",
    "}\n",
    "\n",
    "phrase_map = {\n",
    "    'NT+ST': \"Traffic jam in NT+ST\",\n",
    "    'NS-LEFTS': \"Traffic jam in NS-LEFTS\",\n",
    "    'ET+WT': \"Traffic jam in ET+WT\",\n",
    "    'EW-LEFTS': \"Traffic jam in EW-LEFTS\"\n",
    "}\n",
    "\n",
    "def detect_stalled_vehicle():\n",
    "    root = ET.parse(ROUTE_FILE).getroot()\n",
    "    for v in root.findall(\"vehicle\"):\n",
    "        if v.attrib.get(\"id\") == \"broken\":\n",
    "            for stop in v.findall(\"stop\"):\n",
    "                if int(stop.attrib.get(\"duration\", \"0\")) > 1000:\n",
    "                    lane = stop.attrib.get(\"lane\")\n",
    "                    if \"north\" in lane:\n",
    "                        direction = \"north\"\n",
    "                    elif \"south\" in lane:\n",
    "                        direction = \"south\"\n",
    "                    elif \"east\" in lane:\n",
    "                        direction = \"east\"\n",
    "                    elif \"west\" in lane:\n",
    "                        direction = \"west\"\n",
    "                    else:\n",
    "                        direction = \"unknown\"\n",
    "                    return direction\n",
    "    return None\n",
    "\n",
    "def detect_active_phases(tls_id):\n",
    "    current_lanes = traci.trafficlight.getControlledLanes(tls_id)\n",
    "    state = traci.trafficlight.getRedYellowGreenState(tls_id)\n",
    "    green_lanes = [current_lanes[i] for i in range(len(state)) if state[i] == 'G']\n",
    "    active = {pg: False for pg in phase_groups}\n",
    "    for phase, groups in phase_groups.items():\n",
    "        all_lanes = [l for g in groups for l in lane_groups[g]]\n",
    "        if any(l in green_lanes for l in all_lanes):\n",
    "            active[phase] = True\n",
    "    return active\n",
    "\n",
    "def run_and_detect():\n",
    "    global scenario_flag\n",
    "\n",
    "    # Check for broken vehicle BEFORE running simulation\n",
    "    stalled_direction = detect_stalled_vehicle()\n",
    "    if stalled_direction:\n",
    "        print(f\" Generate optimized traffic signal plan for a broken car near {stalled_direction.lower()} left turn intersection. Scenario #10\")\n",
    "        scenario_flag = 1\n",
    "       \n",
    "        return  # Skip simulation entirely\n",
    "\n",
    "    # Else: proceed with simulation for malfunction or burst\n",
    "    traci.start([\n",
    "        SUMO_BINARY, \"-c\", CFG_FILE,\n",
    "        \"--step-length\", str(STEP),\n",
    "        \"--no-step-log\", \"true\",\n",
    "        \"--time-to-teleport\", \"-1\"\n",
    "    ])\n",
    "\n",
    "    tls_id = traci.trafficlight.getIDList()[0]\n",
    "    next_cut = CYCLE_SEC\n",
    "    starved_flag = 0\n",
    "    occupancy_history = {grp: [] for grp in lane_groups}\n",
    "    active_this_cycle = {pg: False for pg in phase_groups}\n",
    "\n",
    "    while traci.simulation.getMinExpectedNumber() > 0:\n",
    "        traci.simulationStep()\n",
    "        t = traci.simulation.getTime()\n",
    "\n",
    "        for grp, lanes in lane_groups.items():\n",
    "            occs = [traci.lane.getLastStepOccupancy(ln) for ln in lanes]\n",
    "            occupancy_history[grp].append(sum(occs) / len(occs) if occs else 0)\n",
    "\n",
    "        current_active = detect_active_phases(tls_id)\n",
    "        for pg in current_active:\n",
    "            if current_active[pg]:\n",
    "                active_this_cycle[pg] = True\n",
    "\n",
    "        if t >= next_cut:\n",
    "            for pg in phase_groups:\n",
    "                if not active_this_cycle[pg]:\n",
    "                    starved_flag = 1\n",
    "                active_this_cycle[pg] = False\n",
    "            next_cut += CYCLE_SEC\n",
    "\n",
    "    traci.close()\n",
    "\n",
    "    avg_occ_per_group = {\n",
    "        grp: sum(v) / len(v) if v else 0\n",
    "        for grp, v in occupancy_history.items()\n",
    "    }\n",
    "\n",
    "    avg_occ_per_phase = {\n",
    "        pg: max(avg_occ_per_group[g] for g in gs)\n",
    "        for pg, gs in phase_groups.items()\n",
    "    }\n",
    "\n",
    "    most_congested = max(avg_occ_per_phase, key=avg_occ_per_phase.get)\n",
    "    congestion_percent = avg_occ_per_phase[most_congested] * 100\n",
    "\n",
    "    if starved_flag:\n",
    "        print(f\" {phrase_map[most_congested]} due to signal failure, estimated congestion: {int(congestion_percent)}%\")\n",
    "        scenario_flag = 2\n",
    "    else:\n",
    "        print(f\" High congestion detected in {most_congested} at {int(congestion_percent)}%\")\n",
    "        scenario_flag = 3\n",
    "\n",
    "   \n",
    "\n",
    "# Run the full check\n",
    "run_and_detect()\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1a91bef8",
   "metadata": {},
   "source": [
    "printed message will go to llm"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1ddfe864",
   "metadata": {},
   "source": [
    "LLM generated logic "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "74878746",
   "metadata": {},
   "outputs": [],
   "source": [
    "import xml.etree.ElementTree as ET\n",
    "additional = ET.Element(\"additional\")\n",
    "tl_logic = ET.SubElement(additional, \"tlLogic\", {\n",
    "    \"id\": \"center\",\n",
    "    \"type\": \"static\",\n",
    "    \"programID\": \"sudden_et+wt\",\n",
    "    \"offset\": \"0\"\n",
    "})\n",
    "ET.SubElement(tl_logic, \"phase\", {\"duration\": \"50\", \"state\": \"rrrrrGGGGGrrrrrGGGGG\"})\n",
    "ET.SubElement(tl_logic, \"phase\", {\"duration\": \"10\", \"state\": \"rrrrryyyyyrrrrryyyyy\"})\n",
    "ET.SubElement(tl_logic, \"phase\", {\"duration\": \"20\", \"state\": \"GGGGGrrrrrGGGGGrrrrr\"})\n",
    "ET.SubElement(tl_logic, \"phase\", {\"duration\": \"10\", \"state\": \"yyyyyrrrrryyyyyrrrrr\"})\n",
    "tree = ET.ElementTree(additional)\n",
    "\n",
    "\n",
    "tree.write(r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\testing_3_scenarios\\add2.xml\", encoding=\"utf-8\", xml_declaration=True)\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "91b15417",
   "metadata": {},
   "source": [
    "solution using llm generated logic using flag variable flag variable tells which scenario is happening now just run this code "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "13753735",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "🚦 Running simulation for Scenario 3 (Sudden Burst)\n",
      "✅ Simulation complete for Scenario 3 (Sudden Burst).\n"
     ]
    }
   ],
   "source": [
    "# Create a dispatcher that executes a different solution depending on the scenario_flag value\n",
    "\n",
    "\n",
    "import os\n",
    "import xml.etree.ElementTree as ET\n",
    "import traci\n",
    "from sumolib import checkBinary\n",
    "\n",
    "# Predefined config paths\n",
    "SUMO_BINARY = checkBinary(\"sumo-gui\")\n",
    "CFG_FILE = r\"C:\\\\Users\\\\tasfi\\\\Downloads\\\\New folder\\\\Finish_by_today\\\\Finish_by_today\\\\testing_3_scenarios\\\\sumo_config.sumocfg\"\n",
    "ROUTE_FILE = r\"C:\\\\Users\\\\tasfi\\\\Downloads\\\\New folder\\\\Finish_by_today\\\\Finish_by_today\\\\testing_3_scenarios\\\\route.rou.xml\"\n",
    "ADDITIONAL_FILE = \"add2.xml\" #for sudden burst of vehicle  xml name can be different , check with the llm generated code what name it produces write that name here \n",
    "END_TIME = 2300\n",
    "SWITCH_TIME = 800\n",
    "BROKEN_VEHICLE_ID = \"broken\"\n",
    "STEP = 1\n",
    "CYCLE_SEC = 90\n",
    "\n",
    " # change this based on detection phase\n",
    "\n",
    "# Common lane/phase maps\n",
    "lane_groups = {\n",
    "    'NT': ['north_to_center_0', 'north_to_center_1'],\n",
    "    'ST': ['south_to_center_0', 'south_to_center_1'],\n",
    "    'N-L': ['north_to_center_2'],\n",
    "    'S-L': ['south_to_center_2'],\n",
    "    'ET': ['east_to_center_0', 'east_to_center_1'],\n",
    "    'WT': ['west_to_center_0', 'west_to_center_1'],\n",
    "    'E-L': ['east_to_center_2'],\n",
    "    'W-L': ['west_to_center_2']\n",
    "}\n",
    "\n",
    "phase_groups = {\n",
    "    'NS': ['NT', 'ST'],\n",
    "    'NS-LEFTS': ['N-L', 'S-L'],\n",
    "    'EW': ['ET', 'WT'],\n",
    "    'EW-LEFTS': ['E-L', 'W-L']\n",
    "}\n",
    "\n",
    "phrase_map = {\n",
    "    'NS': \"Traffic jam in NT+ST\",\n",
    "    'NS-LEFTS': \"Traffic jam in NS-LEFTS\",\n",
    "    'EW': \"Traffic jam in ET+WT\",\n",
    "    'EW-LEFTS': \"Traffic jam in EW-LEFTS\"\n",
    "}\n",
    "\n",
    "# ------------------------ SCENARIO 1: STALLED VEHICLE ------------------------\n",
    "def add_tllogic_to_config(config_path, tllogic_filename=\"generated_broken_wleft.add.xml\"):  #for stalled vehicle  xml name can be different check with the llm generated code what name it produces write that name here\n",
    "    tree = ET.parse(config_path)\n",
    "    root = tree.getroot()\n",
    "    input_node = root.find(\"input\")\n",
    "    if input_node is None:\n",
    "        input_node = ET.SubElement(root, \"input\")\n",
    "    additional_node = input_node.find(\"additional-files\")\n",
    "    if additional_node is not None:\n",
    "        files = additional_node.get(\"value\").split(\",\")\n",
    "        if tllogic_filename not in files:\n",
    "            files.append(tllogic_filename)\n",
    "            additional_node.set(\"value\", \",\".join(files))\n",
    "    else:\n",
    "        ET.SubElement(input_node, \"additional-files\", value=tllogic_filename)\n",
    "    tree.write(config_path)\n",
    "\n",
    "def run_solution_stalled():\n",
    "    add_tllogic_to_config(CFG_FILE)\n",
    "    traci.start([\n",
    "        SUMO_BINARY, \"-c\", CFG_FILE,\n",
    "        \"--begin\", \"0\", \"--end\", str(END_TIME),\n",
    "        \"--collision.action\", \"none\",\n",
    "        \"--time-to-teleport\", \"-1\",\n",
    "        \"--no-warnings\"\n",
    "    ])\n",
    "    switch_done = False\n",
    "    travel_times, waiting_times = {}, {}\n",
    "\n",
    "    while True:\n",
    "        traci.simulationStep()\n",
    "        sim_time = traci.simulation.getTime()\n",
    "        if sim_time >= SWITCH_TIME and not switch_done:\n",
    "            if BROKEN_VEHICLE_ID in traci.vehicle.getIDList():\n",
    "                print(f\"🚓 {sim_time:.2f}s: Towing vehicle '{BROKEN_VEHICLE_ID}'\")\n",
    "                traci.vehicle.remove(BROKEN_VEHICLE_ID)\n",
    "            print(f\"⏱️ {sim_time:.2f}s: Switching TLS to 'faulty_EL'\")\n",
    "            traci.trafficlight.setProgram(\"center\", \"faulty_EL\")\n",
    "            switch_done = True\n",
    "\n",
    "        for vid in traci.simulation.getDepartedIDList():\n",
    "            travel_times[vid] = {\"depart\": sim_time, \"wait\": 0}\n",
    "        for vid in traci.simulation.getArrivedIDList():\n",
    "            if vid in travel_times:\n",
    "                travel_times[vid][\"arrive\"] = sim_time\n",
    "        for vid in traci.vehicle.getIDList():\n",
    "            if vid in travel_times:\n",
    "                if traci.vehicle.getSpeed(vid) < 0.1:\n",
    "                    travel_times[vid][\"wait\"] += 1\n",
    "        if sim_time >= END_TIME or traci.simulation.getMinExpectedNumber() == 0:\n",
    "            break\n",
    "\n",
    "    traci.close()\n",
    "    print(\"✅ Simulation complete for Scenario 1 (Stalled Vehicle).\")\n",
    "\n",
    "# ------------------------ SCENARIO 2: SIGNAL MALFUNCTION ------------------------\n",
    "def add_tlogic_to_config(sumo_cfg_path, add_file=\"tlLogic_output.add.xml\"):   #signal malfunction\n",
    "    tree = ET.parse(sumo_cfg_path)\n",
    "    root = tree.getroot()\n",
    "    input_elem = root.find(\"input\")\n",
    "    if input_elem is None:\n",
    "        input_elem = ET.SubElement(root, \"input\")\n",
    "    additional = None\n",
    "    for child in input_elem:\n",
    "        if child.tag == \"additional-files\":\n",
    "            additional = child\n",
    "            break\n",
    "    if additional is None:\n",
    "        ET.SubElement(input_elem, \"additional-files\", value=add_file)\n",
    "    else:\n",
    "        current = additional.get(\"value\")\n",
    "        if add_file not in current:\n",
    "            additional.set(\"value\", f\"{current} {add_file}\")\n",
    "    tree.write(sumo_cfg_path)\n",
    "\n",
    "def run_solution_malfunction():\n",
    "    add_tlogic_to_config(CFG_FILE)\n",
    "    traci.start([\n",
    "        SUMO_BINARY, \"-c\", CFG_FILE,\n",
    "        \"--step-length\", str(STEP),\n",
    "        \"--no-step-log\", \"true\",\n",
    "        \"--time-to-teleport\", \"-1\"\n",
    "    ])\n",
    "    tls_id = traci.trafficlight.getIDList()[0]\n",
    "    occupancy_history = {grp: [] for grp in lane_groups}\n",
    "    active_this_cycle = {pg: False for pg in phase_groups}\n",
    "    next_cut = CYCLE_SEC\n",
    "    starved_flag = 0\n",
    "\n",
    "    while traci.simulation.getMinExpectedNumber() > 0:\n",
    "        traci.simulationStep()\n",
    "        t = traci.simulation.getTime()\n",
    "        for grp, lanes in lane_groups.items():\n",
    "            occs = [traci.lane.getLastStepOccupancy(ln) for ln in lanes]\n",
    "            occupancy_history[grp].append(sum(occs) / len(occs) if occs else 0)\n",
    "        current_active = detect_active_phases(tls_id)\n",
    "        for pg in current_active:\n",
    "            if current_active[pg]:\n",
    "                active_this_cycle[pg] = True\n",
    "        if t >= next_cut:\n",
    "            for pg in phase_groups:\n",
    "                if not active_this_cycle[pg]:\n",
    "                    starved_flag = 1\n",
    "                active_this_cycle[pg] = False\n",
    "            next_cut += CYCLE_SEC\n",
    "    traci.close()\n",
    "    print(\"✅ Simulation complete for Scenario 2 (Signal Malfunction).\")\n",
    "\n",
    "# ------------------------ SCENARIO 3: SUDDEN BURST ------------------------\n",
    "def update_sumo_config(cfg_path, route_file_name, additional_file_name):\n",
    "    tree = ET.parse(cfg_path)\n",
    "    root = tree.getroot()\n",
    "    for elem in root.findall(\".//input\"):\n",
    "        for child in list(elem):\n",
    "            if child.tag == \"route-files\":\n",
    "                child.set(\"value\", route_file_name)\n",
    "            elif child.tag == \"additional-files\":\n",
    "                elem.remove(child)\n",
    "        add_elem = ET.Element(\"additional-files\")\n",
    "        add_elem.set(\"value\", additional_file_name)\n",
    "        elem.append(add_elem)\n",
    "    tree.write(cfg_path, encoding=\"utf-8\", xml_declaration=True)\n",
    "\n",
    "def run_solution_burst():\n",
    "    update_sumo_config(CFG_FILE, ROUTE_FILE, ADDITIONAL_FILE)\n",
    "    traci.start([\n",
    "        SUMO_BINARY, \"-c\", CFG_FILE,\n",
    "        \"--step-length\", str(STEP),\n",
    "        \"--no-step-log\", \"true\",\n",
    "        \"--time-to-teleport\", \"-1\"\n",
    "    ])\n",
    "    print(\"🚦 Running simulation for Scenario 3 (Sudden Burst)\")\n",
    "    while traci.simulation.getMinExpectedNumber() > 0:\n",
    "        traci.simulationStep()\n",
    "    traci.close()\n",
    "    print(\"✅ Simulation complete for Scenario 3 (Sudden Burst).\")\n",
    "\n",
    "# ------------------------ MAIN DISPATCH ------------------------\n",
    "if scenario_flag == 1:\n",
    "    run_solution_stalled()\n",
    "elif scenario_flag == 2:\n",
    "    run_solution_malfunction()\n",
    "elif scenario_flag == 3:\n",
    "    run_solution_burst()\n",
    "else:\n",
    "    print(\"⚠️ No valid scenario_flag set.\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "1c89e0f9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "2\n"
     ]
    }
   ],
   "source": [
    "print(scenario_flag)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "88bc1111",
   "metadata": {},
   "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
}
