{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "c1748537",
   "metadata": {},
   "source": [
    "Faulty traffic lights with 200s of green light strvation in east-west approach"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "e1a4dfa2",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "✅ Replaced <tlLogic> block with faulty traffic signal logic.\n"
     ]
    }
   ],
   "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\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": "markdown",
   "id": "deb26f60",
   "metadata": {},
   "source": [
    "refresh every time when running new train data by copying from network folder to combined folder\n",
    "total 6 target folders/train data"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c4b3bdf3",
   "metadata": {},
   "source": [
    "copy code"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6da9ef0a",
   "metadata": {},
   "outputs": [],
   "source": [
    "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",
    "#other target folders are : \n",
    "# Bellevue_150th_Newport__2017-09-11_17-08-32\n",
    "#Bellevue_150th_Newport__2017-09-11_08-08-31\n",
    "#Bellevue_150th_Newport__2017-09-10_18-08-24\n",
    "#Bellevue_116th_NE12th_2017-09-10_19-08-25\n",
    "#150_NE\n",
    "NETWORK_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\new_road_network_for_malfunction\"\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": "5536cf59",
   "metadata": {},
   "source": [
    "to clean up the route.rou.xml files often it shows error"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fb1859b4",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import xml.etree.ElementTree as ET\n",
    "\n",
    "# Base folder containing the minute folders (1Min, 2Min, etc.)\n",
    "BASE_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\Finish_by_today\\NE12_50\"\n",
    "\n",
    "# Minute subfolders to loop over\n",
    "minute_folders = [\"1Min\", \"2Min\", \"3Min\", \"4Min\", \"5Min\",\"6min\"]\n",
    "\n",
    "# Loop over each minute folder\n",
    "for minute in minute_folders:\n",
    "    minute_path = os.path.join(BASE_FOLDER, minute)\n",
    "    if not os.path.isdir(minute_path):\n",
    "        continue\n",
    "\n",
    "    # Loop over all Video_X subfolders\n",
    "    for subfolder in os.listdir(minute_path):\n",
    "        video_path = os.path.join(minute_path, subfolder)\n",
    "        if not os.path.isdir(video_path) or not subfolder.lower().startswith(\"video\"):\n",
    "            continue\n",
    "\n",
    "        route_file = os.path.join(video_path, \"route.rou.xml\")\n",
    "        if not os.path.exists(route_file):\n",
    "            continue\n",
    "\n",
    "        try:\n",
    "            # Parse the XML and remove duplicate vType elements\n",
    "            tree = ET.parse(route_file)\n",
    "            root = tree.getroot()\n",
    "\n",
    "            seen_ids = set()\n",
    "            for elem in root.findall(\"vType\"):\n",
    "                vid = elem.attrib.get(\"id\")\n",
    "                if vid in seen_ids:\n",
    "                    root.remove(elem)\n",
    "                else:\n",
    "                    seen_ids.add(vid)\n",
    "\n",
    "            # Save the cleaned XML\n",
    "            tree.write(route_file, encoding=\"utf-8\", xml_declaration=True)\n",
    "            print(f\"✅ Cleaned: {route_file}\")\n",
    "\n",
    "        except Exception as e:\n",
    "            print(f\"❌ Failed to process {route_file}: {e}\")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0027dbf0",
   "metadata": {},
   "source": [
    "calculate average occupancy for maximum congested phase\n",
    "run copy code and change target folder to calculate occupancy  for each day's data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f2d4ee7b",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "import os\n",
    "import shutil\n",
    "import xml.etree.ElementTree as ET\n",
    "import traci\n",
    "import pandas as pd\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",
    "NETWORK_FOLDER = r\"C:\\Users\\tasfi\\Downloads\\New folder\\Finish_by_today\\Finish_by_today\\new_road_network_for_malfunction\"\n",
    "OUTPUT_CSV = os.path.join(TARGET_FOLDER, \"traffic_malfunctioning_in_east_for_dataset.csv\")\n",
    "SUMO_BINARY = r\"C:\\Program Files (x86)\\Eclipse\\Sumo\\bin\\sumo.exe\"\n",
    "STEP = 1\n",
    "CYCLE_SEC = 90\n",
    "\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",
    "    'NS': ['NT', 'ST'],\n",
    "    'NS-L': ['N-L', 'S-L'],\n",
    "    'EW': ['ET', 'WT'],\n",
    "    'EW-L': ['E-L', 'W-L']\n",
    "}\n",
    "phrase_map = {\n",
    "    'NS': \"High congestion detected in NT+ST\",\n",
    "    'NS-L': \"High congestion detected in NS-LEFTS\",\n",
    "    'EW': \"High congestion detected in ET+WT\",\n",
    "    'EW-L': \"High congestion detected in EW-LEFTS\"\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",
    "    os.makedirs(combined_dir, exist_ok=True)\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",
    "            video_path = os.path.join(duration_path, folder)\n",
    "            route_src = os.path.join(video_path, \"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 update_sumo_config_route_file(cfg_path, route_file):\n",
    "    tree = ET.parse(cfg_path)\n",
    "    root = tree.getroot()\n",
    "    for elem in root.findall(\".//input\"):\n",
    "        for child in elem:\n",
    "            if child.tag == \"route-files\":\n",
    "                child.set(\"value\", route_file)\n",
    "    tree.write(cfg_path, 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",
    "        update_sumo_config_route_file(cfg_file, route_file)\n",
    "\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",
    "        occupancy_history = {grp: [] for grp in lane_groups}\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",
    "            for grp, lanes in lane_groups.items():\n",
    "                lane_occs = [traci.lane.getLastStepOccupancy(ln) for ln in lanes]\n",
    "                avg_lane_occ = sum(lane_occs) / len(lane_occs) if lane_occs else 0\n",
    "                occupancy_history[grp].append(avg_lane_occ)\n",
    "\n",
    "            for veh_id in traci.vehicle.getIDList():\n",
    "                if veh_id not in depart_times:\n",
    "                    depart_times[veh_id] = t  # vehicle enters the simulation\n",
    "                waiting_times[veh_id] = traci.vehicle.getAccumulatedWaitingTime(veh_id)\n",
    "\n",
    "            if t >= next_cut:\n",
    "                next_cut += CYCLE_SEC\n",
    "\n",
    "        # Get arrival time for each vehicle\n",
    "        for veh_id in traci.simulation.getArrivedIDList():\n",
    "            arrival_times[veh_id] = traci.simulation.getTime()\n",
    "\n",
    "        traci.close()\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",
    "        avg_occ_per_group = {\n",
    "            grp: sum(values) / len(values) if values else 0\n",
    "            for grp, values in occupancy_history.items()\n",
    "        }\n",
    "\n",
    "        avg_occ_per_phase = {\n",
    "            phase: max(avg_occ_per_group[g] for g in groups)\n",
    "            for phase, groups in phase_groups.items()\n",
    "        }\n",
    "\n",
    "        most_congested_phase = max(avg_occ_per_phase, key=avg_occ_per_phase.get)\n",
    "        max_value = avg_occ_per_phase[most_congested_phase] * 100\n",
    "        message = phrase_map[most_congested_phase]\n",
    "\n",
    "        print(f\"\\n✅ Simulation {route_file} ended\")\n",
    "        print(f\"📈 {message} at {int(max_value)}%\")\n",
    "        print(f\"⏱️ ATT: {ATT:.2f} s | AWT: {AWT:.2f} s\")\n",
    "\n",
    "        results.append({\n",
    "            \n",
    "            \"congested_phase\": most_congested_phase,\n",
    "            \"max_occupancy_percent\": int(max_value),\n",
    "         \n",
    "        })\n",
    "\n",
    "    return results\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "day_path = os.path.join(BASE_DIR, TARGET_FOLDER)\n",
    "all_results = []\n",
    "for subfolder in os.listdir(day_path):\n",
    "    if subfolder.lower() in [\"2min\", \"3min\", \"4min\", \"5min\", \"6min\"]:\n",
    "        try:\n",
    "            combined_dir = move_routes_and_setup_combined(day_path, subfolder)\n",
    "            cfg_path = os.path.join(combined_dir, \"sumo_config.sumocfg\")\n",
    "            results = run_simulation(cfg_path, combined_dir)\n",
    "            for r in results:\n",
    "                r[\"duration_folder\"] = subfolder\n",
    "            all_results.extend(results)\n",
    "        except Exception as e:\n",
    "            print(f\"❌ Error processing {subfolder}: {e}\")\n",
    "\n",
    "csv_path = os.path.join(TARGET_FOLDER, f\"traffic_malfunctioning_in_east_for_dataset.csv\")\n",
    "results_df = pd.DataFrame(all_results)\n",
    "results_df.to_csv(csv_path, index=False, float_format=\"%.2f\")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5bfb197a",
   "metadata": {},
   "source": [
    "Faulty traffic lights with 200s of green light strvation in north-south approach"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "52d869d3",
   "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\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": "89c7dcdb",
   "metadata": {},
   "source": [
    "evrything else are same\n",
    "repeat the same process \n",
    "copy  code \n",
    "clean up\n",
    "calculate average occupancy for maximum congested phase\n",
    "run copy code each time and change target folder to calculate occupancy  for each day's data\n",
    "save the csv path as \"traffic_malfunctioning_in_north_for_dataset.csv\"\n"
   ]
  }
 ],
 "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
}
