import random
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

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


    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 tick > Config.initialWaitTicks and self.smartCar:  # as we ignore the first 1000 ticks for this
            # add a route to the global registry 计算到达的车辆数
            CarRegistry.totalTrips += 1
            # add the duration for this route to the global tripAverage 路由持续时间
            durationForTrip = (tick - self.currentRouteBeginTick)
            print("CarRegistry.totalTripAverag:",CarRegistry.totalTripAverage)
            CarRegistry.totalTripAverage = addToAverage(CarRegistry.totalTrips,  # 已完成的trips数量 # 100 for faster updates
                                                        CarRegistry.totalTripAverage,   # 所有trip时间的平均值average of all trip durations
                                                        durationForTrip)    # 路由持续时间
            # ((1.0 * totalTrips * totalTripAverage) + durationForTrip) / (totalTrips + 1)
            # CSVLogger.logEvent("arrived", [tick, self.sourceID, self.targetID,
            #                                durationForTip, self.id,self.currentRouterResult.isVictim])
            # log the overrhead values
            # 最小开销 cost_func = lambda u, v, e, prev_e: e['length'] / e['maxSpeed']
            minimalCosts = CustomRouter.minimalRoute(self.sourceID, self.targetID, None, None).totalCost
            # print(minimalCosts)
            # 这个minimalCost有可能起点终点一样就为零了 这里就加了一个判断
            if minimalCosts != 0:
                tripOverhead = durationForTrip / minimalCosts / 1.1  # 1.6 is to correct acceleration and deceleration
            else:
                tripOverhead = durationForTrip / 30 / 1.1
            # when the distance is very short, we have no overhead  距离较短时，没有开销
            if durationForTrip < 10:
                tripOverhead = 1
            # in rare cases a trip does take very long - as most outliers are <30, we cap the overhead to 30 here
            # 在极少数情况下，一次旅行确实需要很长时间——因为大多数异常值都小于30，我们在这里将开销限制为30
            if tripOverhead > 30:
                # print(minimalCosts)
                print("-> capped overhead to 30 - " + str(minimalCosts) + " - " + str(durationForTrip) + " - " + str(
                    tripOverhead))
                tripOverhead = 30

            CarRegistry.totalTripOverheadAverage = addToAverage(CarRegistry.totalTrips,     # 已完成的trips数量
                                                                CarRegistry.totalTripOverheadAverage,   # 所有行程开销的平均值
                                                                tripOverhead)       #
            CSVLogger.logEvent("overhead", [tick, self.sourceID, self.targetID, durationForTrip,
                                            minimalCosts, tripOverhead, self.id, self.currentRouterResult.isVictim])
            # tick 结束时间
            # self.sourceID 起点
            # self.targetID 终点
            # durationForTrip 持续时间
            # minimalCosts 最小开销
            # tripOverhead 实际开销
            # self.id 车辆id
            # self.currentRouterResult.isVictim 是否为被用于探索的route
            # log to kafka
            msg = dict()
            msg["tick"] = tick
            msg["overhead"] = tripOverhead
            RTXForword.publish(msg, Config.kafkaTopicTrips)
        # if car is still enabled, restart it in the simulation
        if self.disabled is False:
            self.addToSimulation(tick)

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

        # if self.targetID is None:
        #     self.sourceID = random.choice(Network.nodes).getID()
        # else:
        #     self.sourceID = self.targetID  # We start where we stopped

        # random target

        # self.targetID = random.choice(Network.nodes).getID()

        # 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)
        self.currentRouterResult = CustomRouter.route(self.sourceID, self.targetID, tick, self, vehicle_data) # reture

        # print(self.currentRouterResult)
        # if len(self.currentRouterResult.route) > 0:
        #     traci.route.add(self.currentRouteID, self.currentRouterResult.route)
        #     # set color to red
        #     return self.currentRouteID
        # else:
        #     # recursion aka. try again as this should work!
        #     return self.__createNewRoute(tick)

    def processTick(self, tick, vehicle_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)
        # print(self.recent_road_id)
        # if self.lastRerouteCounter  >= CustomRouter.reRouteEveryTicks and CustomRouter.reRouteEveryTicks > 0:
        # 如果在路口就重新修改路径
        if ":" in self.recent_road_id and self.recent_road_id != self.old_road_id:
            self.lastRerouteCounter = 0
            if self.smartCar:
                # try:
                #     print("1111111111111111111111111111111111")
                #     # 这里的初始化根本没有用到
                #     # oldRoute = self.currentRouterResult.route
                #     oldRoute = traci.vehicle.getRoute(self.id)
                #     # 获取当前车辆所在的道路ID和下一条道路id
                #     currentEdgeID = traci.vehicle.getRoadID(self.id)
                #     # print(currentEdgeID)
                #     nextNodeID = Network.getEdgeIDsToNode(currentEdgeID).getID()
                #     # print(nextNodeID)
                #     # 调用CustomRouter.route方法来获取新的路径
                #     self.currentRouterResult = CustomRouter.route(nextNodeID, self.targetID, tick, self, vehicle_data)
                #     # 添加了一个newRoute
                #     newRoute = [currentEdgeID] + self.currentRouterResult.route + [self.targetEdgeID]
                #     # newRoute = self.currentRouterResult.route
                #     # info(str(newRoute) + " " + self.id)
                #     traci.vehicle.setRoute(self.id, newRoute)
                #     print("OLD: " + str(oldRoute) + " - NEW: " + str(newRoute) + " id: " + str(self.id))
                # except IndexError as e:
                #     print(e)
                #     pass
                # except traci.exceptions.TraCIException as e:
                #     print(e)
                #     pass
                # except Exception as e:
                #     print(e)
                try:
                    # print("1111111111111111111111111111111111")
                    # 这里的初始化根本没有用到
                    # oldRoute = self.currentRouterResult.route
                    oldRoute = traci.vehicle.getRoute(self.id)
                    # 获取当前车辆所在的道路ID和下一条道路id
                    currentEdgeID = traci.vehicle.getRoadID(self.id)
                    # print("当前车辆道路id为",currentEdgeID)
                    # print(currentEdgeID)
                    # nextNodeID = Network.getEdgeIDsToNode(currentEdgeID).getID()
                    nextNodeID = Network.getEdgeIDsToNode(self.old_road_id).getID()
                    # print(nextNodeID)
                    # 调用CustomRouter.route方法来获取新的路径
                    self.currentRouterResult = CustomRouter.route(nextNodeID, self.targetID, tick, self, vehicle_data)
                    print(self.currentRouterResult)
                    # 添加了一个newRoute
                    newRoute = [currentEdgeID] + self.currentRouterResult.route + [self.targetEdgeID]
                    # newRoute = self.currentRouterResult.route
                    # info(str(newRoute) + " " + self.id)
                    traci.vehicle.setRoute(self.id, newRoute)
                    print("OLD: " + str(oldRoute) + " - NEW: " + str(newRoute) + " id: " + str(self.id))
                except IndexError as e:
                    print(e)
                    pass
                except traci.exceptions.TraCIException as e:
                    print(e)
                    pass
                except Exception as e:
                    print(e)
            #
            try:
                # 车辆当前所在的道路roadID
                roadID = traci.vehicle.getSubscriptionResults(self.id)[80]
                # print(roadID)
                # print(self.currentEdgeID)
                if roadID != self.currentEdgeID and self.smartCar:
                    if self.currentEdgeBeginTick is not None:
                        # 尝试计算一条边需要多长时间
                        CustomRouter.applyEdgeDurationToAverage(self.currentEdgeID, tick - self.currentEdgeBeginTick, tick)
                        # CSVLogger.logEvent("edge", [tick, self.currentEdgeID,
                        #                             tick - self.currentEdgeBeginTick, self.id])
                        # log to kafak
                        # msg = dict()
                        # msg["tick"] = tick
                        # msg["edgeID"] = self.currentEdgeID,
                        # msg["duration"] = tick - self.currentEdgeBeginTick
                    # print("changed route to: " + roadID)
                    self.currentEdgeBeginTick = tick
                    self.currentEdgeID = roadID
                    pass
            except Exception as e:
                print("roadID error " + str(self.id) + " " + str(e))

        self.old_road_id = self.recent_road_id


    def addToSimulation(self, tick, vehicle_data):
        """ adds this car to the simulation through the traci API """
        self.currentRouteBeginTick = tick   # 车辆开始时间
        # try:
        #     # traci.vehicle.addLegacy(self.id, self.__createNewRoute(tick), tick)
        #     self.__createNewRoute(tick, vehicle_data)
        #     # traci.vehicle.add(self.id, self.__createNewRoute(tick), tick, -4, -3)
        #     # 订阅车辆信息
        #     traci.vehicle.subscribe(self.id, (tc.VAR_ROAD_ID,))
        #     # ! currently disabled for performance reasons
        #     # traci.vehicle.setAccel(self.id, self.acceleration)
        #     # traci.vehicle.setDecel(self.id, self.deceleration)
        #     # traci.vehicle.setImperfection(self.id, self.imperfection)
        #     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
        #
        # except Exception as e:
        #     print("error adding " + str(self.id) + " " + str(e))
        #     # try recursion, as this should normally work
        #     # self.addToSimulation(tick)

        # traci.vehicle.addLegacy(self.id, self.__createNewRoute(tick), tick)
        self.__createNewRoute(tick, vehicle_data)
        # traci.vehicle.add(self.id, self.__createNewRoute(tick), tick, -4, -3)
        # 订阅车辆信息
        traci.vehicle.subscribe(self.id, (tc.VAR_ROAD_ID,))
        # ! currently disabled for performance reasons
        # traci.vehicle.setAccel(self.id, self.acceleration)
        # traci.vehicle.setDecel(self.id, self.deceleration)
        # traci.vehicle.setImperfection(self.id, self.imperfection)
        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)
