import random

import torch
import traci
import traci.constants as tc
from app import Config

from app.Util import addToAverage
from app.logdata import CSVLogger
from app.network.Network import Network
from app.routing.CustomRouter import CustomRouter
from app.routing.RouterResult import RouterResult
from app.streaming import RTXForword
from app.logdata import info
from colorama import Fore
import math


class Car:
    """ a abstract class of something that is driving around in the streets """

    def __init__(self, id):
        # the string id
        self.id = id  # type: str
        # the rounds this car already drove
        self.rounds = 0  # type: int
        # the current route as a RouterResult
        self.currentRouterResult = None  # type: RouterResult
        # when we started the route
        self.currentRouteBeginTick = None
        # the id of the current route (somu)
        self.currentRouteID = None  # type: str
        # the id of the current edge/street the car is driving (sumo)
        self.currentEdgeID = None
        # the tick this car got on this edge/street
        self.currentEdgeBeginTick = None
        # the target edge this car drives to
        self.targetEdgeID = None
        # the source edge this car is coming from
        self.sourceEdgeID = None
        # the target node this car drives to
        self.targetID = None
        # the source node this car is coming from
        self.sourceID = None
        # if it is disabled, it will stop driving
        self.disabled = False
        # the cars acceleration in the simulation
        self.acceleration = max(1, random.gauss(4, 2))
        # the cars deceleration in the simulation
        self.deceleration = max(1, random.gauss(6, 2))
        # the driver imperfection in handling the car
        self.imperfection = min(0.9, max(0.1, random.gauss(0.5, 0.5)))
        # is this car a smart car
        # self.smartCar = Config.smartCarPercentage > random.random()
        self.smartCar = True
        # number of ticks since last reroute / arrival
        self.lastRerouteCounter = 0

        # 用于判断车辆是否以及到达下一路段
        self.valid_road = None
        # 是否已经完成过路径重构
        self.is_reroute = False

        # self.x = 0

    def setArrived(self, tick):
        """ car arrived at its target, so we add some statistic data
            汽车到达目标，所以我们添加一些统计数据
        """
        # import here because python can not handle circular-dependencies
        from app.entitiy.CarRegistry import CarRegistry
        # add a round to the car
        self.rounds += 1
        self.lastRerouteCounter = 0
        # 已经运行完一次,所以设置车辆为已到达 disabled = True
        self.disabled = True
        if self.disabled is False:
            self.addToSimulation(tick)

    def __createNewRoute(self, tick, cost_data, car_id_map,is_RL):
        """ creates a new route to a random target and uploads this route to SUMO
            为车辆生成一个新的路径，并将其上传到 SUMO 中
        """
        # import here because python can not handle circular-dependencies



        # self.currentRouteID = self.id + "-" + str(self.rounds)
        # traci.vehicle.getRoute 获取车辆当前路径的起始和终止边缘（edge）ID
        # 获取起始和终止边缘所连接的节点（node）ID
        self.sourceEdgeID = traci.vehicle.getRoute(self.id)[0]
        self.sourceID = Network.getEdgeIDsToNode(self.sourceEdgeID).getID()
        self.targetEdgeID = traci.vehicle.getRoute(self.id)[-1]
        self.targetID = Network.getEdgeIDsFromNode(self.targetEdgeID).getID()
        self.currentRouteID = traci.vehicle.getRouteID(self.id)
        # print(self.id)
        # print(self.sourceEdgeID)
        # print(self.targetEdgeID)
        # print(self.currentRouteID)
        # # 创建车辆ID映射字典
        car_index = car_id_map[self.id]

        cost_data_one = {key: value[car_index] for key, value in cost_data.items()}
        # print("cost_data_one",self.id,cost_data_one)
        self.currentRouterResult = CustomRouter.route(self.sourceEdgeID, self.targetEdgeID, tick, self, cost_data_one,is_RL)
        traci.vehicle.setRoute(self.id, self.currentRouterResult.nodeList)




    def processTick(self, tick, cost_data):
        """ process changes that happened in the tick to this car
            这辆车在运行过程中发生的变化
        """
        self.lastRerouteCounter += 1
        # reroute every x ticks based on config value
        # 根据配置重新路由
        # 获取当前车辆的位置
        self.recent_road_id = traci.vehicle.getRoadID(self.id)  # 车辆所在路段

        if ":" not in self.recent_road_id and self.recent_road_id != '':  # 确保车辆在路段上而不是在交叉路口
            self.recent_lane_id = traci.vehicle.getLaneID(self.id)  # 车辆所在车道
            self.recentCar_pos = traci.vehicle.getPosition(self.id)  # 当前车辆经纬度
            self.laneEnd_pos = traci.lane.getShape(self.recent_lane_id)[-1]  # 车辆所在车道末端位置
            # # 计算车辆到所在车道末端的距离
            self.distance_Car_Lane = math.sqrt(
                (self.recentCar_pos[0] - self.laneEnd_pos[0]) ** 2 + (self.recentCar_pos[1] - self.laneEnd_pos[1]) ** 2)
            # print("self.distance_Car_Lane", self.id, self.distance_Car_Lane - self.x)
            # self.x = self.distance_Car_Lane
            # self.distance_Car_Lane = 1000000
        else:  # 车辆在路口上面
            self.distance_Car_Lane = 1000000

        if self.recent_road_id != self.valid_road:
            self.is_reroute = False

        if self.is_reroute == False and self.distance_Car_Lane < -1:

            self.lastRerouteCounter = 0
            if self.smartCar:
                # 这里的初始化根本没有用到
                # oldRoute = self.currentRouterResult.route
                # print(self.recent_road_id)
                oldRoute = traci.vehicle.getRoute(self.id)
                newRoute = list(oldRoute)[list(oldRoute).index(self.recent_road_id):]

                traci.vehicle.setRoute(self.id, newRoute)

            self.is_reroute = True
        self.valid_road = self.recent_road_id

    def addToSimulation(self, tick, cost_data, car_id_map,is_RL):
        """ adds this car to the simulation through the traci API """
        self.currentRouteBeginTick = tick  # 车辆开始时间
        self.__createNewRoute(tick, cost_data, car_id_map,is_RL)
        # traci.vehicle.add(self.id, self.__createNewRoute(tick), tick, -4, -3)
        # 订阅车辆信息
        traci.vehicle.subscribe(self.id, (tc.VAR_ROAD_ID,))

        if self.smartCar:
            # set color to red
            if self.currentRouterResult.isVictim:
                traci.vehicle.setColor(self.id, (0, 255, 0, 0))
                # print("vehicle " + self.id + "is Victim")
            else:
                traci.vehicle.setColor(self.id, (255, 0, 0, 0))
                # print("vehicle " + self.id + " not Victim")
        else:
            # dump car is using SUMO default routing, so we reroute using the same target
            # putting the next line left == ALL SUMO ROUTING
            traci.vehicle.changeTarget(self.id, self.currentRouterResult.route[-1])
            # 因为这儿原本就有路 所以不管
            pass

    def remove(self):
        """" removes this car from the sumo simulation through traci """
        traci.vehicle.remove(self.id)
