from random import gauss, random

import sys
from dijkstar import Graph, find_path

from app.network.Network import Network
from app.routing.RouterResult import RouterResult

from app.datacollection.Datacollection import DataCollection


class CustomRouter(object):
    """ our own custom defined router """

    # Empty starting references
    edgeMap = None
    graph = None

    # the percentage of smart cars that should be used for exploration 应用于探索的智能汽车的百分比
    explorationPercentage = 0.0  # INITIAL JSON DEFINED!!!
    # randomizes the routes 随机化路线
    routeRandomSigma = 0.2  # INITIAL JSON DEFINED!!!
    # how much speed influences the routing 速度在多少程度上影响了路由
    maxSpeedAndLengthFactor = 1  # INITIAL JSON DEFINED!!!
    # multiplies the average edge value 乘以平均边缘值
    averageEdgeDurationFactor = 1  # INITIAL JSON DEFINED!!!
    # how important it is to get new data 去取得新的数据的重要性
    freshnessUpdateFactor = 10  # INITIAL JSON DEFINED!!!
    # defines what is the oldest value that is still a valid information 定义仍然是有效信息的最古老的值
    freshnessCutOffValue = 500.0  # INITIAL JSON DEFINED!!!
    # how often we reroute cars 我们多久重新路由汽车
    reRouteEveryTicks = 1  # INITIAL JSON DEFINED!!!

    @classmethod
    def init(self):
        """ set up the router using the already loaded network """
        self.graph = Graph()
        self.edgeMap = {}
        for edge in Network.routingEdges:
            # print(f"=============={edge}==============")
            self.edgeMap[edge.id] = edge
            self.graph.add_edge(edge.fromNodeID, edge.toNodeID,
                                {'length': edge.length, 'maxSpeed': edge.maxSpeed,
                                 'lanes': len(edge.lanes), 'edgeID': edge.id})
        # 每个node及其可到达的相邻node，及其所对应的egde
        # print(self.graph)

    @classmethod
    def minimalRoute(cls, fr, to, tick, car):
        # 最小开销
        """creates a minimal route based on length / speed  """
        cost_func = lambda u, v, e, prev_e: e['length'] / e['maxSpeed']
        route = find_path(cls.graph, fr, to, cost_func=cost_func)
        return RouterResult(route, False)

    @classmethod
    def route(cls, fr, to, tick, car, vehicle_data):
        """ creates a route from the f(node) to the t(node) """
        # 1) SIMPLE COST FUNCTION 最大开销 routeRandomSigma 0.2
        # cost_func = lambda u, v, e, prev_e: max(0,gauss(1, CustomRouter.routeRandomSigma) \
        #                                         * (e['length']) / (e['maxSpeed']))

        # if car. :
        #     # here we reduce the cost of an edge based on how old our information is
        #     print("victim routing!")
        #     cost_func = lambda u, v, e, prev_e: (
        #         cls.getAverageEdgeDuration(e["edgeID"]) -
        #         (tick - (cls.edgeMap[e["edgeID"]].lastDurationUpdateTick)) 平均到达时间 减去 信息的衰老程度
        #     )
        # else:
        # 2) Advanced cost function that combines duration with averaging
        # isVictim = ??? random x percent (how many % routes have been victomized before)
        isVictim = cls.explorationPercentage > random()
        if isVictim:
            victimizationChoice = 1
        else:
            victimizationChoice = 0
        """
        ``cost_func``
        A function to apply to each edge to modify its base cost.
        The arguments it will be passed are the current node, 
        a neighbor of the current node, the edge that connects the current node to that neighbor, 
        and the edge that was previously traversed to reach the current node.
        应用于每条边以修改其基本代价的函数。
        传递给它的参数包括当前节点、当前节点的邻居、连接当前节点和邻居的边，以及到达当前节点之前经过的边。
        """
        # 找到边e的路阻值

        # cost_func = lambda u, v, e, prev_e: \
        #     cls.getFreshness(e["edgeID"], tick) * \
        #     cls.averageEdgeDurationFactor * \
        #     cls.getAverageEdgeDuration(e["edgeID"]) \
        #     + \
        #     (1 - cls.getFreshness(e["edgeID"], tick)) * \
        #     cls.maxSpeedAndLengthFactor * \
        #     max(1, gauss(1, cls.routeRandomSigma) *
        #         (e['length']) / e['maxSpeed']) \
        #     - \
        #     (1 - cls.getFreshness(e["edgeID"], tick)) * \
        #     cls.freshnessUpdateFactor * \
        #     victimizationChoice


        cost_func = lambda u, v, e, prev_e: \
            cls.road_cost(e["edgeID"], tick, vehicle_data)

        # generate route 路径生成
        route = find_path(cls.graph, fr, to, cost_func=cost_func)

        # print(f"+++++++++++++++{vehicle_data}+++++++++++++++")

        # print(all_round)
        # Simulation.round = Simulation.round + 1
        # print(f"第{Simulation.round}轮")
        # wrap the route in a result object
        # print(route)
        # print(RouterResult(route, isVictim))
        return RouterResult(route, isVictim)

    @classmethod
    def road_cost(cls, edgeID, tick, vehicle_data):
        # for row in vehicle_data:
        #     if row[0] == tick and row[1] == edgeID:
        #         target_row = row
        #         break
        # print(target_row)
        # print(f"++++++++++++++++++{tick}+++++++++++++++++++")
        # print(vehicle_data[0][0])

        # if tick == vehicle_data[0][1]:
        #     target_row = [row for row in vehicle_data if row[1] == tick and row[0] == str(edgeID)][0]
        #     # print(f"===================={target_row}======================")
        #     # print(vehicle_data[0])
        #     # print(tick)
        #     # print(str(edgeID))
        #     # print(f"+++++++++{target_row}++++++++++++++")
        #     # return 1
        # else:
        #     target_row = [row for row in vehicle_data if row[1] == tick - 1 and row[0] == str(edgeID)][0]
        #     # print(f"===================={target_row}======================")
        #     # print(vehicle_data[0])
        #     # print(tick)
        #     # print(str(edgeID))
        #     # print(f"+++++++++{target_row}++++++++++++++")
        value = vehicle_data[edgeID]
        return value[1] + value[2] + value[3]

    @classmethod
    def getFreshness(cls, edgeID, tick):
        try:
            # 计算现在的tick-上一次更新时的tick的差 来计算多久没更新了
            # lastDurationUpdateTick: 上一次更新时间
            lastUpdate = float(tick) - cls.edgeMap[edgeID].lastDurationUpdateTick
            # 如果超过500步就不新鲜了 相除的分母值会越大 最终的新鲜度的值会越小
            # freshnessCutOffValue = 500
            return 1 - min(1, max(0, lastUpdate / cls.freshnessCutOffValue))
        except TypeError as e:
            # print("error in getFreshnessFactor" + str(e))
            return 1

    @classmethod
    def getAverageEdgeDuration(cls, edgeID):
        """ returns the average duration for this edge in the simulation """
        try:
            print(f"=================={edgeID}===========================")
            return cls.edgeMap[edgeID].averageDuration
        except:
            print("error in getAverageEdgeDuration")
            return 1

    @classmethod
    def applyEdgeDurationToAverage(cls, edge, duration, tick):
        """ tries to calculate how long it will take for a single edge
            试着计算一条边需要多长时间
        """

        try:
            cls.edgeMap[edge].applyEdgeDurationToAverage(duration, tick)
        except:
            return 1
