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


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 = 20 # 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:
            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})

    @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):
        """ 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 = 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

        # generate route
        route = find_path(cls.graph, fr, to, cost_func=cost_func)
        # wrap the route in a result object
        return RouterResult(route, isVictim)

    @classmethod
    def getFreshness(cls, edgeID, tick):
        try:
            # 计算现在的tick-上一次更新时的tick的差 来计算多久没更新了
            lastUpdate = float(tick) - cls.edgeMap[edgeID].lastDurationUpdateTick
            # 如果超过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:
            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
