import json
import traci
import traci.constants as tc
import sumolib
import csv

from app.streaming import RTXForword
from colorama import Fore

from app import Config

from app.entitiy.CarRegistry import CarRegistry
from app.logdata import info
from app.routing.CustomRouter import CustomRouter
from app.streaming import RTXConnector
import time

# get the current system time
from app.routing.RoutingEdge import RoutingEdge
from app.entitiy.Car import Car

from app.datacollection.Datacollection import DataCollection
from app.network.Network import Network

from app.logdata import CSVLogger

current_milli_time = lambda: int(round(time.time() * 1000))

class Simulation(object):
    """ here we run the simulation in """

    # the current tick of the simulation
    tick = 0

    # last tick time
    lastTick = current_milli_time()

    road_data = []

    # net = sumolib.net.readNet(Config.sumoNet)



    @classmethod
    def applyFileConfig(cls):
        """ reads configs from a json and applies it at realtime to the simulation """
        try:
            config = json.load(open('./knobs.json'))
            # "the percentage of routes we want to use for exploration"
            # 我们想要用于探索的路由百分比
            CustomRouter.explorationPercentage = config['explorationPercentage']
            # "how much the averageEdgeFactor influences the routing"
            # 路段edge因子对路由的影响有多大
            CustomRouter.averageEdgeDurationFactor = config['averageEdgeDurationFactor']
            # "how much the length/speed influences the routing"
            # 长度/速度对路由的影响程度
            CustomRouter.maxSpeedAndLengthFactor = config['maxSpeedAndLengthFactor']
            # "how much the freshnessUpdateFactor influences the routing"
            # freshnessUpdateFactor对路由的影响程度
            CustomRouter.freshnessUpdateFactor = config['freshnessUpdateFactor']
            # "if data is older than this we do not consider it in the algorithm"
            # 如果数据早于此，我们就不考虑它在算法中
            CustomRouter.freshnessCutOffValue = config['freshnessCutOffValue']
            # 在汽车启动后寻找新路线每x次，多久更新一次路段
            # "check for a new route very x times after the car starts"
            CustomRouter.reRouteEveryTicks = config['reRouteEveryTicks']
        except:
            pass

    @classmethod
    def start(cls):
        """ start the simulation """
        info("# Start adding initial cars to the simulation", Fore.MAGENTA)
        # apply the configuration from the json file    json文件加载
        cls.applyFileConfig()
        info(Fore.GREEN + "# apply the configuration from the json file OK !" +  Fore.RESET)
        # CarRegistry.applyCarCounter()
        info(Fore.GREEN + "# start loop !" + Fore.RESET)
        cls.loop()




    @classmethod
    # @profile
    def loop(cls):
        """ loops the simulation """

        # start listening to all cars that arrived at their target
        # 订阅所有到达终点的车辆的ID
        traci.simulation.subscribe((tc.VAR_ARRIVED_VEHICLES_IDS,))
        loopOver = False
        data_all = []

        Network.loadNetwork()
        edge_ids = list(Network.edgeIds)
        while not loopOver:
            # Do one simulation step
            cls.tick += 1
            traci.simulationStep()
            print(f"===================={cls.tick}=========================")
            # # 获取当前tick车道信息

            if cls.tick == 1:
                vehicle_data = DataCollection.get_orig_data(cls.tick, edge_ids)
            else:
                vehicle_data = DataCollection.mod_data(cls.tick, vehicle_data, edge_ids)
            # print(vehicle_data)
            # print(vehicle_data)

            list_vehicle_data = []
            for key, value in vehicle_data.items():
                list_vehicle_data.append([key] + value)
            data_all = data_all + list_vehicle_data

            # 检查SUMO仿真中的车辆列表，并将未注册的车辆添加到车辆注册表中，并进行仿真
            CarRegistry.applyCarCounter(cls.tick - 1, vehicle_data)

            # Log tick duration to kafka
            # 计算每个tick持续的时间 记录到kafka
            duration = current_milli_time() - cls.lastTick
            # print(duration)
            cls.lastTick = current_milli_time()
            msg = dict()
            msg["duration"] = duration
            RTXForword.publish(msg, Config.kafkaTopicPerformance)

            # Check for removed cars and re-add them into the system
            # 检查是否有被删除的车辆，并将这些车辆重新添加到系统中
            # 目的是确保模拟中所有的车辆都能被正确处理，即使它们在某些情况下被意外地从模拟中删除了。
            # 通过重新添加这些车辆并将它们的到达时间设置为当前tick
            # print(traci.simulation.getSubscriptionResults()[122])
            for removedCarId in traci.simulation.getSubscriptionResults()[122]:
                # 它使用CarRegistry类的findById方法找到相应的Car对象，并将其到达时间设置为当前tick
                # print(CarRegistry.findById(removedCarId))
                # 可以确保程序不会因为缺少车辆对象而崩溃。在这种情况下，使用NullCar.setArrived对象来替代真实的车辆对象，以便在不影响其他部分的情况下继续进行处理。
                CarRegistry.findById(removedCarId).setArrived(cls.tick)

            # "crowd-nav-routing"中"duration"的意义是？
            timeBeforeCarProcess = current_milli_time()
            # let the cars process this step
            CarRegistry.processTick(cls.tick, vehicle_data)
            # log time it takes for routing 记录路由所需要的时间
            msg = dict()
            msg["duration"] = current_milli_time() - timeBeforeCarProcess

            RTXForword.publish(msg, Config.kafkaTopicRouting)

            # if we enable this we get debug information in the sumo-gui using global traveltime
            # should not be used for normal running, just for debugging
            # if (cls.tick % 10) == 0:
            # for e in Network.routingEdges:
            # 1)     traci.edge.adaptTraveltime(e.id, 100*e.averageDuration/e.predictedDuration)
            #     traci.edge.adaptTraveltime(e.id, e.averageDuration)
            # 3)     traci.edge.adaptTraveltime(e.id, (cls.tick-e.lastDurationUpdateTick)) # how old the data is

            # real time update of config if we are not in kafka mode
            # 实时配置更新
            if (cls.tick % 10) == 0:
                if Config.kafkaUpdates is False and Config.mqttUpdates is False:
                    # json mode  json模式
                    print("使用json配置")
                    cls.applyFileConfig()

                else:
                    # kafka mode  kafka模式
                    print("使用kafka配置")
                    newConf = RTXConnector.checkForNewConfiguration()
                    # newConf == None
                    if newConf is not None:
                        if "exploration_percentage" in newConf:
                            CustomRouter.explorationPercentage = newConf["exploration_percentage"]
                            print("setting victimsPercentage: " + str(newConf["exploration_percentage"]))
                        if "route_random_sigma" in newConf:
                            CustomRouter.routeRandomSigma = newConf["route_random_sigma"]
                            print("setting routeRandomSigma: " + str(newConf["route_random_sigma"]))
                        if "max_speed_and_length_factor" in newConf:
                            CustomRouter.maxSpeedAndLengthFactor = newConf["max_speed_and_length_factor"]
                            print("setting maxSpeedAndLengthFactor: " + str(newConf["max_speed_and_length_factor"]))
                        if "average_edge_duration_factor" in newConf:
                            CustomRouter.averageEdgeDurationFactor = newConf["average_edge_duration_factor"]
                            print("setting averageEdgeDurationFactor: " + str(newConf["average_edge_duration_factor"]))
                        if "freshness_update_factor" in newConf:
                            CustomRouter.freshnessUpdateFactor = newConf["freshness_update_factor"]
                            print("setting freshnessUpdateFactor: " + str(newConf["freshness_update_factor"]))
                        if "freshness_cut_off_value" in newConf:
                            CustomRouter.freshnessCutOffValue = newConf["freshness_cut_off_value"]
                            print("setting freshnessCutOffValue: " + str(newConf["freshness_cut_off_value"]))
                        if "re_route_every_ticks" in newConf:
                            CustomRouter.reRouteEveryTicks = newConf["re_route_every_ticks"]
                            print("setting reRouteEveryTicks: " + str(newConf["re_route_every_ticks"]))
                        if "total_car_counter" in newConf:
                            CarRegistry.totalCarCounter = newConf["total_car_counter"]
                            CarRegistry.applyCarCounter()
                            print("setting totalCarCounter: " + str(newConf["total_car_counter"]))
                        if "edge_average_influence" in newConf:
                            RoutingEdge.edgeAverageInfluence = newConf["edge_average_influence"]
                            print("setting edgeAverageInfluence: " + str(newConf["edge_average_influence"]))

            # print status update if we are not running in parallel mode
            # 如果不是在并行模式下运行，则打印状态更新
            if (cls.tick % 1) == 0 and Config.parallelMode is False:
                print(str(Config.processID) + " -> Step:" + str(cls.tick) + " # Driving cars: " + str(
                    traci.vehicle.getIDCount()) + "/" + str(
                    CarRegistry.totalCarCounter) + " # avgTripDuration: " + str(
                    CarRegistry.totalTripAverage) + "(" + str(
                    CarRegistry.totalTrips) + ")" + " # avgTripOverhead: " + str(
                    CarRegistry.totalTripOverheadAverage))
            # Config.processID  进程号
            # cls.tick  仿真步骤
            # traci.vehicle.getIDCount()  当前正在运行的车辆数量
            # CarRegistry.totalCarCounter   车辆总数
            # # avgTripDuration = CarRegistry.totalTripAverage + CarRegistry.totalTrips
            # # avgTripOverhead = CarRegistry.totalTripOverheadAverage

                # @depricated -> will be removed
                # # if we are in paralllel mode we end the simulation after 10000 ticks with a result output
                # if (cls.tick % 10000) == 0 and Config.parallelMode:
                #     # end the simulation here
                #     print(str(Config.processID) + " -> Step:" + str(cls.tick) + " # Driving cars: " + str(
                #         traci.vehicle.getIDCount()) + "/" + str(
                #         CarRegistry.totalCarCounter) + " # avgTripDuration: " + str(
                #         CarRegistry.totalTripAverage) + "(" + str(
                #         CarRegistry.totalTrips) + ")" + " # avgTripOverhead: " + str(
                #         CarRegistry.totalTripOverheadAverage))
                #     return




            # 判断循环是否结束
            loopOver = True
            for key in CarRegistry.cars:
                if not CarRegistry.cars[key].disabled:
                    loopOver = False
                    break

            # # 判断循环是否结束
            # if cls.tick == 3:
            #     loopOver = True

        # print(data_all)
        # 写入csv
        # CSVLogger.logEvent("tick_data_111", data_all)


        with open('./app/data/tickdata_'+str(Config.formatted_datetime)+'.csv', mode='w', newline='') as file:
            writer = csv.writer(file)
            writer.writerows(data_all)


