#!/usr/bin/env python import copy from datetime import datetime, timedelta from aman.config.RunwaySequencing import RunwayAssignmentType from aman.sys.aco.Configuration import Configuration from aman.sys.aco.Constraints import SpacingConstraints from aman.sys.aco.Node import Node class RunwayManager: def __init__(self, configuration : Configuration): self.Spacings = SpacingConstraints() self.Configuration = configuration self.RunwayInbounds = copy.deepcopy(configuration.PreceedingRunwayInbounds) self.IafInbounds = copy.deepcopy(configuration.PreceedingIafInbounds) def calculateEarliestArrivalTime(self, runway : str, node : Node, earliestArrivalTime : datetime): constrainedETA = None if None != self.RunwayInbounds[runway]: # get the WTC based ETA if None == self.RunwayInbounds[runway].Inbound.WTC or None == node.Inbound.WTC: spacingWTC = 3 else: if self.RunwayInbounds[runway].Inbound.WTC not in self.Spacings.WtcSpacing: spacingWTC = 3 elif node.Inbound.WTC not in self.Spacings.WtcSpacing[self.RunwayInbounds[runway].Inbound.WTC]: spacingWTC = self.Spacings.WtcSpacing[self.RunwayInbounds[runway].Inbound.WTC]['L'] else: spacingWTC = self.Spacings.WtcSpacing[self.RunwayInbounds[runway].Inbound.WTC][node.Inbound.WTC] # get the runway time spacing spacingRunway = self.Configuration.RunwayConstraints.findRunway(runway).Spacing constrainedETA = self.RunwayInbounds[runway].Inbound.PlannedArrivalTime + timedelta(minutes = max(spacingWTC, spacingRunway) / (node.Inbound.PerformanceData.SpeedApproach / 60)) # calculate the arrival times for the dependent inbounds for dependentRunway in self.Configuration.RunwayConstraints.findDependentRunways(runway): if None != self.RunwayInbounds[dependentRunway.Runway.Name]: candidate = self.RunwayInbounds[dependentRunway.Runway.Name].Inbound.PlannedArrivalTime + timedelta(minutes = 3 / (node.Inbound.PerformanceData.SpeedApproach / 60)) if None == constrainedETA or candidate > constrainedETA: constrainedETA = candidate if None == constrainedETA: eta = max(node.ArrivalCandidates[runway].InitialArrivalTime, earliestArrivalTime) else: eta = max(node.ArrivalCandidates[runway].InitialArrivalTime, max(constrainedETA, earliestArrivalTime)) return eta, eta - node.ArrivalCandidates[runway].InitialArrivalTime def selectShallShouldMayArrivalRunway(self, node : Node, runways, earliestArrivalTime : datetime): candidate = None delay = None for runway in runways: eta, _ = self.calculateEarliestArrivalTime(runway.Runway.Name, node, earliestArrivalTime) if None == delay: delay = eta - node.ArrivalCandidates[runway.Runway.Name].InitialArrivalTime candidate = runway elif delay > (eta - node.ArrivalCandidates[runway.Runway.Name].InitialArrivalTime): delay = eta- node.ArrivalCandidates[runway.Runway.Name].InitialArrivalTime candidate = runway return candidate def executeShallShouldMayAssignment(self, node : Node, earliestArrivalTime : datetime): shallRunways = [] shouldRunways = [] mayRunways = [] for runway in self.Configuration.RunwayConstraints.ActiveArrivalRunways: # test the shall assignments if RunwayAssignmentType.AircraftType in runway.ShallAssignments: if node.Inbound.Report.aircraft.type in runway.ShallAssignments[RunwayAssignmentType.AircraftType]: shallRunways.append(runway) if RunwayAssignmentType.GateAssignment in runway.ShallAssignments: if node.Inbound.Report.plannedGate in runway.ShallAssignments[RunwayAssignmentType.GateAssignment]: shallRunways.append(runway) # test the should assignments if RunwayAssignmentType.AircraftType in runway.ShouldAssignments: if node.Inbound.Report.aircraft.type in runway.ShouldAssignments[RunwayAssignmentType.AircraftType]: shouldRunways.append(runway) if RunwayAssignmentType.GateAssignment in runway.ShouldAssignments: if node.Inbound.Report.plannedGate in runway.ShouldAssignments[RunwayAssignmentType.GateAssignment]: shouldRunways.append(runway) # test the may assignments if RunwayAssignmentType.AircraftType in runway.MayAssignments: if node.Inbound.Report.aircraft.type in runway.MayAssignments[RunwayAssignmentType.AircraftType]: eta, _ = self.calculateEarliestArrivalTime(runway.Runway.Name, node, earliestArrivalTime) if (eta - node.ArrivalCandidates[runway.Runway.Name].InitialArrivalTime) <= self.Configuration.AirportConfiguration.MaxDelayMay: mayRunways.append(runway) if RunwayAssignmentType.GateAssignment in runway.MayAssignments: if node.Inbound.Report.plannedGate in runway.MayAssignments[RunwayAssignmentType.GateAssignment]: eta, _ = self.calculateEarliestArrivalTime(runway.Runway.Name, node, earliestArrivalTime) if (eta - node.ArrivalCandidates[runway.Runway.Name].InitialArrivalTime) <= self.Configuration.AirportConfiguration.MaxDelayMay: mayRunways.append(runway) runway = self.selectShallShouldMayArrivalRunway(node, shallRunways, earliestArrivalTime) if None != runway: return [ runway ] runway = self.selectShallShouldMayArrivalRunway(node, shouldRunways, earliestArrivalTime) if None != runway: return [ runway ] runway = self.selectShallShouldMayArrivalRunway(node, mayRunways, earliestArrivalTime) if None != runway: return [ runway ] return self.Configuration.RunwayConstraints.ActiveArrivalRunways def selectArrivalRunway(self, node : Node, earliestArrivalTime : datetime): availableRunways = self.Configuration.RunwayConstraints.ActiveArrivalRunways if 0 == len(availableRunways): return None, None, None if True == self.Configuration.RunwayConstraints.UseShallShouldMay and None == node.Inbound.RequestedRunway: availableRunways = self.executeShallShouldMayAssignment(node, earliestArrivalTime) elif None != node.Inbound.RequestedRunway: for runway in availableRunways: if node.Inbound.RequestedRunway == runway.Runway.Name: availableRunways = [ runway ] break if 0 == len(availableRunways): runway = self.Configuration.RunwayConstraints.ActiveArrivalRunways[0] eta, delta = self.calculateEarliestArrivalTime(runway.Runway.Name, node, earliestArrivalTime) return runway, eta, delta # start with the beginning selectedRunway = None lostTime = None eta = None # get the runway with the earliest ETA for runway in availableRunways: candidate, delta = self.calculateEarliestArrivalTime(runway.Runway.Name, node, earliestArrivalTime) if None == eta or eta > candidate: selectedRunway = runway.Runway lostTime = delta eta = candidate return selectedRunway, eta, lostTime