benchmarks/JetStream2/RexBench/FlightPlanner/flight_planner.js (1,332 lines of code) (raw):

/* * Copyright (C) 2017 Apple Inc. All rights reserved. * Copyright (C) 2016-2017 Michael Saboff. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ "use strict"; let earthRadius = 3440; // In nautical miles. let TwoPI = Math.PI * 2; let degreeCharacter = "\u00b0"; let regExpOptionalUnicodeFlag; var keywords; if (this.useUnicode) { regExpOptionalUnicodeFlag = "u"; keywords = UnicodeStrings; } else { regExpOptionalUnicodeFlag = ""; keywords = { get: function(str) { return str; } }; } function status(text) { console.debug("Status: " + text); } function error(text) { console.error("Error: " + text); } if (typeof(Number.prototype.toRadians) === "undefined") { Number.prototype.toRadians = function() { return this * Math.PI / 180; } } if (typeof(Number.prototype.toDegrees) === "undefined") { Number.prototype.toDegrees = function() { return this * 180 / Math.PI; } } function distanceFromSpeedAndTime(speed, time) { return speed * time.hours(); } let LatRE = new RegExp("^([NS\\-])?(90|[0-8]?\\d)(?:( [0-5]?\\d\\.\\d{0,3})'?|(\\.\\d{0,6})|( ([0-5]?\\d)\" ?([0-5]?\\d)'?))?", "i" + regExpOptionalUnicodeFlag); function decimalLatitudeFromString(latitudeString) { if (typeof latitudeString != "string") return 0; let match = latitudeString.match(LatRE); if (!match) return 0; let result = 0; let sign = 1; if (match[1] && (match[1].toUpperCase() == "S" || match[1] == "-")) sign = -1; result = Number(match[2]); if (result != 90) { if (match[3]) { // e.g. N37 42.874 let minutes = Number(match[3]); result = result + (minutes / 60); } else if (match[4]) { // e.g. N37.30697 let decimalDegrees = Number(match[4]); result = result + decimalDegrees; } else if (match[5]) { // e.g. N37 18" 27' let degrees = Number(match[6]); let minutes = Number(match[7]); result = result + (degrees + minutes / 60) / 60; } } return result * sign; } let LongRE = new RegExp("^([EW\\-]?)(180|(?:1[0-7]|\\d)?\\d)(?:( [0-5]?\\d\\.\\d{0,3})|(\\.\\d{0,6})|( ([0-5]?\\d)\" ?([0-5]?\\d)'?)?)", "i" + regExpOptionalUnicodeFlag); function decimalLongitudeFromString(longitudeString) { if (typeof longitudeString != "string") return 0; let match = longitudeString.match(LongRE); if (!match) return 0; let result = 0; let sign = 1; if (match[1] && (match[1].toUpperCase() == "W" || match[1] == "-")) sign = -1; result = Number(match[2]); if (result != 180) { if (match[3]) { // e.g. W121 53.254 let minutes = Number(match[3]); result = result + (minutes / 60); } else if (match[4]) { // e.g. W121.8876 let decimalDegrees = Number(match[4]); result = result + decimalDegrees; } else if (match[5]) { // e.g. W121 53" 15' let degrees = Number(match[6]); let minutes = Number(match[7]); result = result + (degrees + minutes / 60) / 60; } } return result * sign; } let TimeRE = new RegExp("^([0-9][0-9]?)(?:\:([0-5][0-9]))?(?:\:([0-5][0-9]))?$"); class Time { constructor(time) { if (time instanceof Date) { this._seconds = Math.Round(time.valueOf() / 1000); return; } if (typeof time == "string") { let match = time.match(TimeRE); if (!match) { this._seconds = 0; return; } if (match[3]) { let hours = parseInt(match[1].toString()); let minutes = parseInt(match[2].toString()); let seconds = parseInt(match[3].toString()); this._seconds = (hours * 60 + minutes) * 60 + seconds; } else if (match[2]) { let minutes = parseInt(match[1].toString()); let seconds = parseInt(match[2].toString()); this._seconds = minutes * 60 + seconds; } else this._seconds = parseInt(match[1].toString()); return; } if (typeof time == "number") { this._seconds = Math.round(time); return; } this._seconds = 0; } add(otherTime) { return new Time(this._seconds + otherTime._seconds); } addDate(otherDate) { return new Date(this._seconds * 1000 + otherDate.valueOf()); } static differenceBetween(time2, time1) { let seconds1; let seconds2; if (time1 instanceof Time) seconds1 = time1.seconds(); else seconds1 = Math.Round(time1.valueOf() / 1000); if (time2 instanceof Time) seconds2 = time2.seconds(); else seconds2 = Math.Round(time2.valueOf() / 1000); return new Time(seconds2 - seconds1); } seconds() { return this._seconds; } minutes() { return this._seconds / 60; } hours() { return this._seconds / 3600; } toString() { let result = ""; let seconds = this._seconds % 60; if (seconds < 0) { result = "-"; seconds = -seconds; } let minutes = this._seconds / 60 | 0; let hours = minutes / 60 | 0; minutes = minutes % 60; if (hours) result = result + hours + ":"; if (minutes < 10 && hours) result = result + "0"; result = result + minutes + ":"; if (seconds < 10) result = result + "0"; result = result + seconds; return result; } } class GeoLocation { constructor(latitude, longitude) { this.latitude = latitude; this.longitude = longitude; } latitudeString() { let latitude = this.latitude; let latitudePrefix = "N"; if (latitude < 0) { latitude = -latitude; latitudePrefix = "S" } let latitudeDegrees = Math.floor(latitude); let latitudeMinutes = ((latitude - latitudeDegrees) * 60).toFixed(3); let latitudeMinutesFiller = latitudeMinutes < 10 ? " " : ""; return latitudePrefix + latitudeDegrees + degreeCharacter + latitudeMinutesFiller + latitudeMinutes + "'"; } longitudeString() { let longitude = this.longitude; let longitudePrefix = "E"; if (longitude < 0) { longitude = -longitude; longitudePrefix = "W" } let longitudeDegrees = Math.floor(longitude); let longitudeMinutes = ((longitude - longitudeDegrees) * 60).toFixed(3); let longitudeMinutesFiller = longitudeMinutes < 10 ? " " : ""; return longitudePrefix + longitudeDegrees + degreeCharacter + longitudeMinutesFiller + longitudeMinutes + "'"; } distanceTo(otherLocation) { let dLat = (otherLocation.latitude - this.latitude).toRadians(); let dLon = (otherLocation.longitude - this.longitude).toRadians(); let a = Math.sin(dLat/2) * Math.sin(dLat/2) + Math.cos(this.latitude.toRadians()) * Math.cos(otherLocation.latitude.toRadians()) * Math.sin(dLon/2) * Math.sin(dLon/2); let c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a)); return earthRadius * c; } bearingFrom(otherLocation, magneticVariation) { if (magneticVariation == undefined) magneticVariation = 0; let dLon = (this.longitude - otherLocation.longitude).toRadians(); let thisLatitudeRadians = this.latitude.toRadians(); let otherLatitudeRadians = otherLocation.latitude.toRadians(); let y = Math.sin(dLon) * Math.cos(this.latitude.toRadians()); let x = Math.cos(otherLatitudeRadians) * Math.sin(thisLatitudeRadians) - Math.sin(otherLatitudeRadians) * Math.cos(thisLatitudeRadians) * Math.cos(dLon); return (Math.atan2(y, x).toDegrees() + 720 + magneticVariation) % 360; } bearingTo(otherLocation, magneticVariation) { if (magneticVariation == undefined) magneticVariation = 0; let dLon = (otherLocation.longitude - this.longitude).toRadians(); let thisLatitudeRadians = this.latitude.toRadians(); let otherLatitudeRadians = otherLocation.latitude.toRadians(); let y = Math.sin(dLon) * Math.cos(otherLocation.latitude.toRadians()); let x = Math.cos(thisLatitudeRadians) * Math.sin(otherLatitudeRadians) - Math.sin(thisLatitudeRadians) * Math.cos(otherLatitudeRadians) * Math.cos(dLon); return (Math.atan2(y, x).toDegrees() + 720 + magneticVariation) % 360 } locationFrom(bearing, distance, magneticVariation) { if (magneticVariation == undefined) magneticVariation = 0; let bearingRadians = (bearing - magneticVariation).toRadians(); let thisLatitudeRadians = this.latitude.toRadians(); let angularDistance = distance / earthRadius; let latitudeRadians = Math.asin(Math.sin(thisLatitudeRadians) * Math.cos(angularDistance) + Math.cos(thisLatitudeRadians) * Math.sin(angularDistance) * Math.cos(bearingRadians)); let longitudeRadians = this.longitude.toRadians() + Math.atan2(Math.sin(bearingRadians) * Math.sin(angularDistance) * Math.cos(thisLatitudeRadians), Math.cos(angularDistance) - Math.sin(thisLatitudeRadians) * Math.sin(latitudeRadians)); return new GeoLocation(latitudeRadians.toDegrees(), longitudeRadians.toDegrees()); } toString() { return "(" + this.latitudeString() + ", " + this.longitudeString() + ")"; } } function findFaaWaypoint(waypoint) { return faaWaypoints[waypoint]; } class FaaWaypoints { constructor() { if (!FaaWaypoints.instance) { FaaWaypoints.instance = this; this.waypoints = _faaWaypoints; } return FaaWaypoints.instance; } find(waypoint) { return this.waypoints[waypoint]; } } FaaWaypoints.instance = undefined; let faaWaypoints = new FaaWaypoints(); class FaaAirways { constructor() { if (!FaaAirways.instance) { FaaAirways.instance = this; this.airways = _faaAirways; } return FaaAirways.instance; } isAirway(identifier) { return !!this.airways[identifier]; } resolveAirway(airwayID, entryPoint, exitPoint) { let airway = this.airways[airwayID]; if (!airway) return ""; let entryIndex = airway.fixes.indexOf(entryPoint); let exitIndex = airway.fixes.indexOf(exitPoint); if (entryIndex == -1 || exitIndex == -1) return ""; let stride = (entryIndex <= exitIndex) ? 1 : -1; let route = []; for (let idx = entryIndex; idx != exitIndex; idx = idx + stride) route.push(airway.fixes[idx]); route.push(airway.fixes[exitIndex]); return route; } } FaaAirways.instance = undefined; let faaAirways = new FaaAirways(); class UserWaypoints { constructor() { if (!UserWaypoints.instance) { UserWaypoints.instance = this; this.waypoints = {}; } return UserWaypoints.instance; } clear() { this.waypoints = {}; } find(waypoint) { return this.waypoints[waypoint]; } update(name, description, latitude, longitude) { if (typeof latitude == "string") latitude = decimalLatitudeFromString(latitude); if (typeof longitude == "string") longitude = decimalLongitudeFromString(longitude); this.waypoints[name.toUpperCase()] = { "name": name, "description": description, "latitude": latitude, "longitude": longitude }; } } UserWaypoints.instance = undefined; let userWaypoints = new UserWaypoints(); class EngineConfig { constructor(type, fuelFlow, trueAirspeed) { this.type = type; this._fuelFlow = fuelFlow; this._trueAirspeed = trueAirspeed; } trueAirspeed() { return this._trueAirspeed; } fuelFlow() { return this._fuelFlow; } static appendConfig(type, fuelFlow, trueAirspeed) { if (this.allConfigsByType[type]) { status("Duplicate Engine configuration: " + type); return; } var newConfig = new EngineConfig(type, fuelFlow, trueAirspeed); this.allConfigs.push(newConfig); this.allConfigsByType[type] = newConfig; } static getConfig(n) { if (n >= this.allConfigs.length) return undefined; return this.allConfigs[n]; } } EngineConfig.allConfigs = []; EngineConfig.allConfigsByType = {}; EngineConfig.Taxi = 0; EngineConfig.Runup = 1; EngineConfig.Takeoff = 2; EngineConfig.Climb = 3; EngineConfig.Cruise = 4; EngineConfig.Pattern= 5; class Waypoint { constructor(name, type, description, latitude, longitude) { this.name = name; this.type = type; this.description = description; this.latitude = latitude; this.longitude = longitude; } } class Leg { constructor(fix, location) { this.previous = undefined; this.next = undefined; this.fix = fix; this.location = location; this.course = 0; this.distance = 0; this.trueAirspeed = 0; this.windDirection = 0; this.windSpeed = 0; this.heading = 0; this.estGS = 0; this.startFlightTiming = false; this.stopFlightTiming = false; this.engineConfig = EngineConfig.Cruise; this.fuelFlow = 0; this.distanceRemaining = 0; this.estimatedTimeEnroute = undefined; this.estTimeRemaining = 0; this.estFuel = 0; } fixName() { return this.fix; } toString() { return this.fix; } setPrevious(leg) { this.previous = leg; } previousLeg() { return this.previous; } setNext(leg) { this.next = leg; } nextLeg() { return this.next; } setWind(windDirection, windSpeed) { this.windDirection = windDirection; this.windSpeed = windSpeed; } isSameWind(windDirection, windSpeed) { return this.windDirection == windDirection && this.windSpeed == windSpeed; } windToString() { if (!this.windSpeed) return ""; return (this.windDirection ? this.windDirection : "360") + "@" + this.windSpeed; } setTrueAirspeed(trueAirspeed) { this.trueAirspeed = trueAirspeed; } isStandardTrueAirspeed() { let engineConfig = EngineConfig.getConfig(this.engineConfig); return (this.trueAirspeed == engineConfig.trueAirspeed()); } trueAirspeedToString() { return this.trueAirspeed + "kts"; } updateDistanceAndBearing(other) { this.distance = this.location.distanceTo(other); this.course = Math.round(this.location.bearingFrom(other)); if (this.estimatedTimeEnroute == undefined && this.estGS != 0) { let estimatedTimeEnrouteInSeconds = Math.round(this.distance * 3600 / this.estGS); this.estimatedTimeEnroute = new Time(estimatedTimeEnrouteInSeconds); } if (this.estimatedTimeEnroute.seconds()) this.estFuel = this.fuelFlow * this.estimatedTimeEnroute.hours(); } propagateWind() { let windDirection = this.windDirection; let windSpeed = this.windSpeed; windDirection = (windDirection + 360) % 360; if (!windDirection) windDirection = 360; for (let currLeg = this; currLeg; currLeg = currLeg.nextLeg()) { currLeg.windDirection = windDirection; currLeg.windSpeed = windSpeed; if (currLeg.stopFlightTiming) break; } } updateForWind() { if (!this.windSpeed || !this.trueAirspeed) { this.heading = this.course; this.estGS = this.trueAirspeed; return; } let windDirectionRadians = this.windDirection.toRadians(); let courseRadians = this.course.toRadians(); let swc = (this.windSpeed / this.trueAirspeed) * Math.sin(windDirectionRadians - courseRadians); if (Math.abs(swc) > 1) { status("Wind to strong to fly!"); return; } let headingRadians = courseRadians + Math.asin(swc); if (headingRadians < 0) headingRadians += TwoPI; if (headingRadians > TwoPI) headingRadians -= TwoPI let groundSpeed = this.trueAirspeed * Math.sqrt(1 - swc * swc) - this.windSpeed * Math.cos(windDirectionRadians - courseRadians); if (groundSpeed < 0) { status("Wind to strong to fly!"); return; } this.estGS = groundSpeed; this.heading = Math.round(headingRadians.toDegrees()); } calculate() { let engineConfig = EngineConfig.getConfig(this.engineConfig); if (!this.trueAirspeed) this.trueAirspeed = engineConfig.trueAirspeed(); this.fuelFlow = engineConfig.fuelFlow(); this.updateForWind(); } updateForward() { if (this.specialUpdateForward) this.specialUpdateForward(); let previousLeg = this.previousLeg(); let havePrevious = true; if (!previousLeg) { havePrevious = false; previousLeg = this; if (!this.estimatedTimeEnroute) this.estimatedTimeEnroute = new Time(0); } let thisLegType = this.type; if (thisLegType == "Climb" && havePrevious) this.location = previousLeg.location; else { this.updateDistanceAndBearing(previousLeg.location); this.updateForWind(); let nextLeg = this.nextLeg(); let previousLegType = previousLeg.type; if (havePrevious) { if (previousLegType == "Climb") { let climbDistance = distanceFromSpeedAndTime(previousLeg.estGS, previousLeg.climbTime); if (climbDistance < this.distance) { let climbStartLocation = previousLeg.location; let climbEndLocation = climbStartLocation.locationFrom(this.course, climbDistance); previousLeg.location = climbEndLocation; previousLeg.updateDistanceAndBearing(climbStartLocation); this.estimatedTimeEnroute = undefined; this.updateDistanceAndBearing(climbEndLocation); } else { status("Not enough distance to climb in leg #" + previousLeg.index); } } else if ((thisLegType == "Left" || thisLegType == "Right") && nextLeg && nextLeg.location) { let standardRateCircumference = this.trueAirspeed / 30; let standardRateRadius = standardRateCircumference / TwoPI; let offsetInboundBearing = 360 + previousLeg.course + (thisLegType == "Left" ? -90 : 90); offsetInboundBearing = Math.round((offsetInboundBearing + 360) % 360); // Save original location if (!previousLeg.originalLocation) previousLeg.originalLocation = previousLeg.location; let previousLocation = previousLeg.originalLocation; let inboundLocation = previousLocation.locationFrom(offsetInboundBearing, standardRateRadius); let bearingToNext = Math.round(nextLeg.location.bearingFrom(previousLocation)); let offsetOutboundBearing = bearingToNext + (thisLegType == "Left" ? 90 : -90); offsetOutboundBearing = (offsetOutboundBearing + 360) % 360; let outboundLocation = previousLocation.locationFrom(offsetOutboundBearing, standardRateRadius); let turnAngle = thisLegType == "Left" ? (360 + bearingToNext - previousLeg.course) : (360 + previousLeg.course - bearingToNext); turnAngle = (turnAngle + 360) % 360; let totalDegrees = turnAngle + 360 * this.extraTurns; let secondsInTurn = Math.round(totalDegrees / 3); this.estimatedTimeEnroute = new Time(Math.round((turnAngle + 360 * this.extraTurns) / 3)); this.estFuel = this.fuelFlow * this.estimatedTimeEnroute.hours(); this.location = outboundLocation; this.distance = distanceFromSpeedAndTime(this.trueAirspeed, this.estimatedTimeEnroute); previousLeg.location = inboundLocation; let prevPrevLeg = previousLeg.previousLeg(); if (prevPrevLeg && prevPrevLeg.location) { previousLeg.estimatedTimeEnroute = undefined; previousLeg.updateDistanceAndBearing(prevPrevLeg.location); } } } } } updateBackward() { let nextLeg = this.nextLeg(); let distanceRemaining; let timeRemaining; if (nextLeg) { distanceRemaining = nextLeg.distanceRemaining; timeRemaining = nextLeg.estTimeRemaining; } else { distanceRemaining = 0; timeRemaining = new Time(0); } if (this.stopFlightTiming || timeRemaining.seconds()) { this.distanceRemaining = distanceRemaining + this.distance;; this.estTimeRemaining = timeRemaining.add(this.estimatedTimeEnroute); } else this.estTimeRemaining = new Time(0); } } let RallyLegWithFixRE = new RegExp("^([0-9a-z\.]{3,16})\\|(" + keywords.get("START") + "|" + keywords.get("TIMING") + ")", "i" + regExpOptionalUnicodeFlag); class RallyLeg extends Leg { constructor(type, fix, location, engineConfig) { super(fix, location); this.type = type; this.engineConfig = engineConfig; } fixName() { return this.type; } toString() { return this.fixName(); } static reset() { RallyLeg.startLocation = undefined; RallyLeg.startFix = ""; RallyLeg.totalTaxiTime = new Time(0); RallyLeg.taxiSegments = []; } static fixNeeded(fix) { let match = fix.match(RallyLegWithFixRE); if (!match) return ""; return match[1].toString(); } static getLegWithFix(waypointText, fix, location) { let match = waypointText.match(RallyLegWithFixRE); if (!match) return undefined; let legType = match[2].toString(); if (legType == keywords.get("START")) { if (this.startLocation) { status("Trying to create second start leg"); return undefined; } this.startLocation = location; this.startFix = fix; this.totalTaxiTime = new Time(0); this.taxiSegments = []; return new StartLeg(waypointText, fix, location); } if (legType == keywords.get("TIMING")) return new TimingLeg(waypointText, fix, location); error("Unhandled Rally Leg type " + legType); return undefined; } } RallyLeg.startLocation = undefined; RallyLeg.startFix = ""; RallyLeg.totalTaxiTime = new Time(0); RallyLeg.taxiSegments = []; class StartLeg extends RallyLeg { constructor(fixText, fix, location) { super("Start", fix, location, EngineConfig.Taxi); } fixName() { return this.fix + "|Start"; } } class TimingLeg extends RallyLeg { constructor(fixText, fix, location) { super("Timing", fix, location, EngineConfig.Cruise); this.stopFlightTiming = true; } fixName() { return this.fix + "|Timing"; } } let RallyLegNoFixRE = new RegExp(keywords.get("TAXI") + "|" + keywords.get("RUNUP") + "|" + keywords.get("TAKEOFF") + "|" + keywords.get("CLIMB") + "|" + keywords.get("PATTERN") + "|" + keywords.get("LEFT") + "|" + keywords.get("RIGHT"), "i" + regExpOptionalUnicodeFlag); class RallyLegWithoutFix extends RallyLeg { constructor(type, CommentsAsFix, location, engineConfig) { super(type, CommentsAsFix, location, engineConfig); } setPrevious(previous) { if (this.setLocationFromPrevious() && previous) this.location = previous.location; super.setPrevious(previous); } setLocationFromPrevious() { return false; } static isRallyLegWithoutFix(fix) { let barPosition = fix.indexOf("|"); let firstPart = barPosition < 0 ? fix : fix.substring(0, barPosition); return RallyLegNoFixRE.test(firstPart); } static getLegNoFix(waypointText) { let barPosition = waypointText.indexOf("|"); let firstPart = barPosition < 0 ? waypointText : waypointText.substring(0, barPosition); firstPart = firstPart.toUpperCase(); let match = firstPart.match(RallyLegNoFixRE); if (!match) return undefined; let legType = match[0].toString(); if (legType == keywords.get("TAXI")) return new TaxiLeg(waypointText); if (legType == keywords.get("RUNUP")) return new RunupLeg(waypointText); if (legType == keywords.get("TAKEOFF")) { if (!this.startLocation) { status("Trying to create a Takeoff leg without start leg"); return undefined; } return new TakeoffLeg(waypointText); } if (legType == keywords.get("CLIMB")) return new ClimbLeg(waypointText); if (legType == keywords.get("PATTERN")) return new PatternLeg(waypointText); if (legType == keywords.get("LEFT") || legType == keywords.get("RIGHT")) return new TurnLeg(waypointText, legType == keywords.get("RIGHT")); error("Unhandled Rally Leg type " + legType); return undefined; } } // TAXI[|<time>] e.g. TAXI|2:30 let TaxiLegRE = new RegExp("^" + keywords.get("TAXI") + "(?:\\|([0-9][0-9]?(?:\:[0-5][0-9])?))?$", "i" + regExpOptionalUnicodeFlag); class TaxiLeg extends RallyLegWithoutFix { constructor(fixText) { let match = fixText.match(TaxiLegRE); super("Taxi", "", new GeoLocation(-1, -1), EngineConfig.Taxi); let taxiTimeString = "5:00"; if (match[1]) taxiTimeString = match[1].toString(); this.estimatedTimeEnroute = new Time(taxiTimeString); } setLocationFromPrevious() { return true; } fixName() { return "Taxi|" + this.estimatedTimeEnroute.toString(); } } // RUNUP[|<time>] e.g. RUNUP|0:30 let RunupLegRE = new RegExp("^" + keywords.get("RUNUP") + "(?:\\|([0-9][0-9]?(?:\:[0-5][0-9])?))?$", "i" + regExpOptionalUnicodeFlag); class RunupLeg extends RallyLegWithoutFix { constructor(fixText) { let match = fixText.match(RunupLegRE); super("Runup", "", new GeoLocation(-1, -1), EngineConfig.Runup); let runupTimeString = "30"; if (match[1]) runupTimeString = match[1].toString(); this.estimatedTimeEnroute = new Time(runupTimeString); } setLocationFromPrevious() { return true; } fixName() { return "Runup|" + this.estimatedTimeEnroute.toString(); } } // TAKEOFF[|<time>][|<bearing>|<distance>] e.g. TAKEOFF|2:00|270@3.5 let TakeoffLegRE = new RegExp("^" + keywords.get("TAKEOFF") + "(?:\\|([0-9][0-9]?(?:\:[0-5][0-9])?))?(?:\\|([0-9]{1,2}|[0-2][0-9][0-9]|3[0-5][0-9]|360)(?:@)(\\d{1,2}(?:\\.\\d{1,4})?))?$", "i" + regExpOptionalUnicodeFlag); class TakeoffLeg extends RallyLegWithoutFix { constructor(fixText) { let match = fixText.match(TakeoffLegRE); let bearingFromStart = 0; let distanceFromStart = 0; let takeoffEndLocation = RallyLeg.startLocation; if (match && match[2] && match[3]) { bearingFromStart = parseInt(match[2].toString()) % 360; distanceFromStart = parseFloat(match[3].toString()); takeoffEndLocation = RallyLeg.startLocation.locationFrom(bearingFromStart, distanceFromStart); } super("Takeoff", "", takeoffEndLocation, EngineConfig.Takeoff); this.bearingFromStart = bearingFromStart; this.distanceFromStart = distanceFromStart; let takeoffTimeString = "2:00"; if (match[1]) takeoffTimeString = match[1].toString(); this.estimatedTimeEnroute = new Time(takeoffTimeString); this.startFlightTiming = true; } fixName() { let result = "Takeoff"; if (this.estimatedTimeEnroute.seconds() != 120) result += "|" + this.estimatedTimeEnroute.toString(); if (this.distanceFromStart) result += "|" + this.bearingFromStart + "@" + this.distanceFromStart; return result; } } // CLIMB|<alt>|<time> e.g. CLIMB|5000|7:00 let ClimbLegRE = new RegExp("^" + keywords.get("CLIMB") + "(?:\\|)(\\d{3,5})(?:\\|([0-9][0-9]?(?:\:[0-5][0-9])?))$", "i" + regExpOptionalUnicodeFlag); class ClimbLeg extends RallyLegWithoutFix { constructor(fixText) { let match = fixText.match(ClimbLegRE); let altitude = 5500; if (match && match[1]) altitude = match[1].toString(); super("Climb", altitude + "\"", undefined, EngineConfig.Climb); let timeToClimb = "8:00"; if (match && match[2]) timeToClimb = match[2].toString(); this.altitude = altitude; this.climbTime = this.estimatedTimeEnroute = new Time(timeToClimb); } setLocationFromPrevious() { return true; } fixName() { return "Climb|" + this.altitude + "|" + this.estimatedTimeEnroute.toString(); } } // PATTERN|<time> e.g. PATTERN|0:30 let PatternLegRE = new RegExp("^" + keywords.get("PATTERN") + "(?:\\|([0-9][0-9]?(?:\:[0-5][0-9])?))$", "i" + regExpOptionalUnicodeFlag); class PatternLeg extends RallyLegWithoutFix { constructor(fixText) { super("Pattern", "", undefined, EngineConfig.Pattern); let match = fixText.match(PatternLegRE); let patternTimeString = match[1].toString(); this.estimatedTimeEnroute = new Time(patternTimeString); } setLocationFromPrevious() { return true; } fixName() { return "Pattern|" + this.estimatedTimeEnroute.toString(); } } // {LEFT,RIGHT}[|+<extra_turns>] e.g. LEFT|2 let TurnLegRE = new RegExp("^(" + keywords.get("LEFT") + "|" + keywords.get("RIGHT") + ")(?:\\|\\+(\\d))?$", "i"); class TurnLeg extends RallyLegWithoutFix { constructor(fixText, isRightTurn) { let match = fixText.match(TurnLegRE); let direction = "Left"; if (match && match[1]) direction = match[1].toString().toUpperCase() == keywords.get("LEFT") ? "Left" : "Right"; let engineConfig = EngineConfig.Cruise; super(direction, "", new GeoLocation(-1, -1), engineConfig); this.extraTurns = (match && match[2]) ? parseInt(match[2]) : 0; } fixName() { let result = this.type; if (this.extraTurns) result += ("|+" + this.extraTurns); return result; } } let LegModifier = new RegExp("(360|3[0-5][0-9]|[0-2][0-9]{2}|[0-9]{1,2})@([0-9]{1,3})|([1-9][0-9]{1,2}|0)kts", "i"); class FlightPlan { constructor(name, route) { this._name = name; this._route = route; this._firstLeg = undefined; this._lastLeg = undefined; this._legCount = 0; this._defaultWindDirection = 0; this._defaultWindSpeed = 0; this._trueAirspeedOverride = 0; this._timeToGate = undefined; this._estimatedTimeEnroute = undefined; this._totalFuel = 0; this._totalTime = new Time(); this._gateTime = new Time(); RallyLeg.reset(); // Refactor to make this more OO } clear() { } appendLeg(leg) { if (!this._firstLeg) this._firstLeg = leg; if (this._lastLeg) this._lastLeg.setNext(leg); leg.setPrevious(this._lastLeg); leg.setNext(undefined); if (this._trueAirspeedOverride) { leg.setTrueAirspeed(this._trueAirspeedOverride); this.clearTrueAirspeedOverride(0); } if (this._defaultWindSpeed) leg.setWind(this._defaultWindDirection, this._defaultWindSpeed); this._lastLeg = leg; this._legCount++; } setDefaultWind(windDirection, windSpeed) { this._defaultWindDirection = windDirection; this._defaultWindSpeed = windSpeed; } clearTrueAirspeedOverride() { this._trueAirspeedOverride = 0; } setTrueAirspeedOverride(trueAirspeed) { this._trueAirspeedOverride = trueAirspeed; } isLegModifier(fix) { return LegModifier.test(fix); } processLegModifier(fix) { let match = fix.match(LegModifier); if (match) { if (match[1] && match[2]) { let windDirection = parseInt(match[1].toString()) % 360; let windSpeed = parseInt(match[2].toString()); this.setDefaultWind(windDirection, windSpeed); } else if (match[3]) { let trueAirspeed = parseInt(match[3].toString()); this.setTrueAirspeedOverride(trueAirspeed); } } } resolveWaypoint(waypointText) { if (this.isLegModifier(waypointText)) this.processLegModifier(waypointText); else if (RallyLegWithoutFix.isRallyLegWithoutFix(waypointText)) { let rallyLeg = RallyLegWithoutFix.getLegNoFix(waypointText); if (rallyLeg) this.appendLeg(rallyLeg); } else { let fixName = RallyLeg.fixNeeded(waypointText); let isRallyWaypoint = false; if (fixName) isRallyWaypoint = true; else fixName = waypointText; let waypoint = userWaypoints.find(fixName); if (!waypoint) waypoint = faaWaypoints.find(fixName); if (!waypoint) { error("Couldn't find waypoint \"" + waypointText + "\""); return; } let location = new GeoLocation(waypoint.latitude, waypoint.longitude); if (isRallyWaypoint) { let rallyLeg = RallyLeg.getLegWithFix(waypointText, fixName, location); this.appendLeg(rallyLeg); } else this.appendLeg(new Leg(waypoint.name, location)); } } parseRoute() { let waypointsToLookup = this._route.split(/ +/); let priorWaypoint = ""; for (let waypointIndex = 0; waypointIndex < waypointsToLookup.length; waypointIndex++) { let currentWaypoint = waypointsToLookup[waypointIndex].toUpperCase(); if (faaAirways.isAirway(currentWaypoint) && (waypointIndex + 1) < waypointsToLookup.length) { let exitWaypointFix = waypointsToLookup[waypointIndex + 1]; let airwayFixes = faaAirways.resolveAirway(currentWaypoint, priorWaypoint, exitWaypointFix); // We skip the entry and exit fixes, because they are handled in the prior / next // iterations of the outer loop. for (let airwayFixIndex = 1; airwayFixIndex < airwayFixes.length - 1; airwayFixIndex++) this.resolveWaypoint(airwayFixes[airwayFixIndex]); } else this.resolveWaypoint(currentWaypoint); priorWaypoint = currentWaypoint; } } calculate() { if (!this._firstLeg) return; let haveStartTiming = false; let haveStopTiming = false; for (let thisLeg = this._firstLeg; thisLeg; thisLeg = thisLeg.nextLeg()) { thisLeg.calculate(); if (thisLeg.startFlightTiming) { if (haveStartTiming) status("Have duplicate Start timing leg in row " + thisLeg.toString()); haveStartTiming = true; } if (thisLeg.stopFlightTiming) { if (haveStopTiming) status("Have duplicate Timing leg in row " + thisLeg.toString()); haveStopTiming = true; } } if (!haveStartTiming) this._firstLeg.startFlightTiming = true; if (!haveStopTiming) this._lastLeg.stopFlightTiming = true; for (let thisLeg = this._firstLeg; thisLeg; thisLeg = thisLeg.nextLeg()) thisLeg.updateForward(); for (let thisLeg = this._lastLeg; thisLeg; thisLeg = thisLeg.previousLeg()) thisLeg.updateBackward(); for (let thisLeg = this._firstLeg; thisLeg; thisLeg = thisLeg.nextLeg()) { if (thisLeg.startFlightTiming) this._gateTime = thisLeg.estTimeRemaining; } this._totalTime = this._firstLeg.estTimeRemaining; } resolvedRoute() { let result = ""; let lastWindDirection = 0; let lastWindSpeed = 0; let legIndex = 0; let currentLeg = this._firstLeg; for (; currentLeg; currentLeg = currentLeg.nextLeg(), legIndex++) { if (legIndex) result = result + " "; if (!currentLeg.isSameWind(lastWindDirection, lastWindSpeed)) { result = result + currentLeg.windToString() + " "; lastWindDirection = currentLeg.windDirection; lastWindSpeed = currentLeg.windSpeed; } if (!currentLeg.isStandardTrueAirspeed()) result = result + currentLeg.trueAirspeedToString() + " "; result = result + currentLeg.toString(); } return result; } name() { return this._name; } totalTime() { return this._totalTime; } gateTime() { return this._gateTime; } toString() { let result = ""; let lastWindDirection = 0; let lastWindSpeed = 0; let legIndex = 0; let currentLeg = this._firstLeg; for (; currentLeg; currentLeg = currentLeg.nextLeg(), legIndex++) { if (legIndex) result = result + " "; if (!currentLeg.isSameWind(lastWindDirection, lastWindSpeed)) { result = result + currentLeg.windToString() + " "; lastWindDirection = currentLeg.windDirection; lastWindSpeed = currentLeg.windSpeed; } if (!currentLeg.isStandardTrueAirspeed()) result = result + currentLeg.trueAirspeedToString() + " "; result = result + currentLeg.toString(); result = result + " " + currentLeg.location + " " + currentLeg.distance.toFixed(2) + "nm " + currentLeg.estGS.toFixed(2) + "kts " + currentLeg.estimatedTimeEnroute + " "; } if (this._gateTime) result = result + " gate time " + this._gateTime; result = result + " total time " + this._firstLeg.estTimeRemaining; return result; } } EngineConfig.appendConfig("Taxi", 2, 0); EngineConfig.appendConfig("Runup", 8, 0); EngineConfig.appendConfig("Takeoff", 27, 105); EngineConfig.appendConfig("Climb", 22, 125); EngineConfig.appendConfig("Cruise", 15, 142); EngineConfig.appendConfig("Pattern", 11, 95); class ExpectedFlightPlan { constructor(name, route, expectedRoute, expectedTotalTime, expectedGateTime) { this._name = name; this._route = route; this._expectedRoute = expectedRoute; this._expectedTotalTime = expectedTotalTime; this._expectedGateTime = expectedGateTime; } reset() { this._flightPlan = new FlightPlan(this._name, this._route); } resolveRoute() { this._flightPlan.parseRoute(); } calculate() { this._flightPlan.calculate(); } checkExpectations() { if (this._expectedRoute) { let computedRoute = this._flightPlan.resolvedRoute(); if (this._expectedRoute != computedRoute) error("Flight plan " + this._flightPlan.name() + " route different than expected (\"" + this._expectedRoute + "\"), got (\"" + computedRoute + "\")"); } if (this._expectedTotalTime) { let computedTotalTime = this._flightPlan.totalTime(); let deltaTime = Math.abs(Time.differenceBetween(this._expectedTotalTime, computedTotalTime).seconds()); if (deltaTime > 5) error("Flight plan " + this._flightPlan.name() + " total time different than expected (" + this._expectedTotalTime + "), got (" + computedTotalTime + ")"); } if (this._expectedGateTime) { let computedGateTime = this._flightPlan.gateTime(); let deltaTime = Math.abs(Time.differenceBetween(this._expectedGateTime, computedGateTime).seconds()); if (deltaTime > 5) error("Flight plan " + this._flightPlan.name() + " gate time different than expected (" + this._expectedGateTime + "), got (" + computedGateTime + ")"); } } } function setupUserWaypoints() { userWaypoints.clear(); userWaypoints.update("Oilcamp", "Oil storage in the middle of no where", "36.68471", "-120.50277"); userWaypoints.update("I5.Westshields", "I5 & West Shields", "36.77774", "-120.72426"); userWaypoints.update("I5.165", "Intersection of I5 and CA165", "36.93022", "-120.84068"); userWaypoints.update("I5.VOLTA", "I5 & Volta Road", "37.01419", "-120.92878"); userWaypoints.update("PT.ALPHA", "Intersection of I5 and CA152", "37.05665", "-120.96990"); userWaypoints.update("Jellysferry", "Jelly's Ferry bridge across Sacramento River", "N40 19.037", "W122 11.359"); userWaypoints.update("Howie", "RDD Timing Point", "N40 21.893", "W122 13.042"); userWaypoints.update("Hale", "2014 Leg 1 Timing", "N39 33.621", "W119 14.438"); userWaypoints.update("Winnie", "2014 Leg 2 Timing", "N40 50.499", "W114 12.595"); userWaypoints.update("WindRiver", "2014 Leg 3 Timing", "N42 43.733", "W108 38.800"); userWaypoints.update("Buff", "2014 Leg 4 Timing", "N43 59.455", "W103 16.171"); userWaypoints.update("Omega", "2014 Leg 5 Timing", "N44 53.388", "W95 38.935"); userWaypoints.update("Paul", "2014 Leg 6 Timing", "N43 22.027", "W89 37.111"); userWaypoints.update("MicrowaveSt", "2014 Microwave Station", "N40 24.89", "W117 12.37"); userWaypoints.update("RanchTower", "2014 Ranch/Tower", "N41 6.16", "W115 5.43"); userWaypoints.update("FremontIsland", "2014 Fremont Island", "N41 10.49", "W112 20.64"); userWaypoints.update("Tremonton", "2014 Tremonton", "N41 42.86", "W112 11.05"); userWaypoints.update("RandomPoint", "2014 Random Point", "N42", "W111 03"); userWaypoints.update("Farson", "2014 Farson", "N42 6.40", "W109 26.95"); userWaypoints.update("Midwest", "2014 Midwest", "N43 24.49", "W109 16.68"); userWaypoints.update("Bill", "2014 Bill", "N43 13.96", "W105 15.60"); userWaypoints.update("MMNHS", "2014 MMNHS", "N43 52.67", "W101 57.65"); userWaypoints.update("Tracks", "2014 Tracks", "N44 21", "W100 22"); userWaypoints.update("Towers", "2014 Towers", "N43 34.25", "W92 25.64"); userWaypoints.update("IsletonBridge", "Isleton Bridge", "N38 10.32", "W121 35.62"); userWaypoints.update("Mystery15", "2015 Mystery", "N38 46.22", "W122 34.25"); userWaypoints.update("Paskenta", "Paskenta Town", "N39 53.13", "W122 32.36"); userWaypoints.update("Bonanza", "Bonanza Town", "N42 12.15", "W121 24.53"); userWaypoints.update("Silverlake", "Silverlake", "N43 07.41", "W121 03.74"); userWaypoints.update("Millican", "Bend Timing Start", "N43 52.75", "W120 55.13"); userWaypoints.update("Goering", "Bend Timing", "N44 05.751", "W120 56.834"); userWaypoints.update("Constantia2", "Our Constantia Wpt", "N39 56.068", "W120 0.831"); userWaypoints.update("Hallelujah2", "Reno Timing", "N39 46.509", "W120 2.336"); userWaypoints.update("Redding.Pond", "Pond 6nm North of KRDD", "N40 36", "W122 17"); userWaypoints.update("Thunderhill", "Thunder Hill Race Track", "N39 32.36", "W122 19.83"); userWaypoints.update("CascadeHighway", "Cascade Wonderland Highway", "N40 46.63", "W122 19.12"); userWaypoints.update("Eagleville", "Eagleville closed airport", "N41 18.73", "W120 3.00"); userWaypoints.update("DuckLakePass", "Saddle near Duck Lake", "N41 3.00", "W120 3.00"); } function createTestRoutes() { let flightPlans = [ { name: "Rally Practice 1", route: "C83|Start Taxi|6:00 Runup|0:30 Taxi|1:00 Takeoff Climb|5500|5:00 CA67 0CN1 28CA 126kts I5.165 PT.ALPHA KLSN|Timing Pattern|0:45 Taxi|2:00" }, { name: "Rally Practice 2", route: "C83|Start Taxi|6:00 Runup|0:30 Taxi|1:00 Takeoff Climb|5500|5:00 ECA 343@4 5CL3 67CA 314@12 126kts I5.165 126kts PT.ALPHA|Timing 126kts KLSN Pattern|0:45 Taxi|2:00" }, { name: "Rally Practice 3", route: "KTCY|Start Taxi|6:00 Runup|0:30 Taxi|1:00 Takeoff Climb|3500|5:00 O27 67CA OILCAMP I5.WESTSHIELDS I5.165 I5.VOLTA PT.ALPHA|Timing KLSN Pattern|0:45 Taxi|2:00" }, { name: "Rally Practice 4", route: "C83|Start Taxi|6:00 Runup|0:30 Taxi|1:00 Takeoff Climb|5500|5:00 62@3 9CL0 342@9 CL84 28CA 315@10 126kts I5.165 126kts PT.ALPHA|Timing 126kts KLSN Pattern|0:45 Taxi|2:00" }, { name: "Rally Practice 5", route: "C83|Start Taxi|6:00 Runup|0:30 Taxi|1:00 Takeoff Climb|5500|7:00 O27 9CL0 CL01 I5.165 PT.ALPHA KLSN Pattern|0:45 Taxi|2:00" }, { name: "2014 HWD Rally Leg 1", route: "KHWD|Start Taxi|6:00 Runup|0:30 Taxi|1:00 Takeoff Climb|5500|4:35 VPDUB Climb|8500|6:25 2CL9 Left|+1 09CL Left|+1 O02 Left|+1 77NV Left 126kts HALE|Timing KSPZ Pattern|0:45 Taxi|2:00"}, { name: "2014 HWD Rally Leg 2", route: "KSPZ|Start Taxi|4:00 Runup|0:30 Taxi|1:00 Takeoff Climb|7500|9:00 NV30 Left MicrowaveSt Left DOBYS Left RanchTower Left 126kts Winnie|Timing KENV Pattern|0:45 Taxi|2:00"}, { name: "2014 HWD Rally Leg 3", route: "KENV|Start Taxi|5:00 Runup|0:30 Taxi|1:00 Takeoff Climb|7500|8:30 FremontIsland Left|+1 Tremonton Left|+1 RandomPoint Left|+1 Farson Left 126kts WindRiver|Timing KLND Pattern|0:45 Taxi|2:00"}, { name: "2014 HWD Rally Leg 4", route: "KLND|Start Taxi|6:00 Runup|0:30 Taxi|1:00 Takeoff Climb|7500|10:00 Midwest Left Bill Right|+1 WY09 Left KCUT Left 126kts BUFF|Timing KRAP Pattern|0:45 Taxi|2:00"}, { name: "2014 HWD Rally Leg 5", route: "KRAP|Start Taxi|5:00 Runup|0:30 Taxi|3:00 Takeoff Climb|7500|9:00 MMNHS Left|+1 Tracks Left|+1 8D7 Left 5H3 Left 126kts Omega KMVE Pattern|0:45 Taxi|2:00"}, { name: "2014 HWD Rally Leg 6", route: "KMVE|Start Taxi|6:00 Runup|0:30 Taxi|2:00 Takeoff Climb|5500|6:00 1D6 Left 68Y Right|+1 Towers Left KCHU Left 126kts Paul|Timing KMSN Pattern|0:45 Taxi|2:00"}, { name: "2015 HWD Rally Leg 1", route: "KHWD|Start Taxi|6:00 Runup|0:30 Taxi|1:00 Takeoff Climb|5500|4:35 VPDUB Climb|5500|3:25 IsletonBridge Mystery15 Left|+1 71CL Left|+1 Paskenta Left|+1 O37 Left|+1 JELLYSFERRY HOWIE|Timing KRDD Pattern|0:45 Taxi|2:00"}, { name: "2015 HWD Rally Leg 2", route: "KRDD|Start Taxi|6:00 Runup|0:30 Taxi|2:00 Takeoff Climb|8500|1:20 REDDING.POND Climb|8500|7:40 A26 O81 Left 340@6 Bonanza Left|+1 Silverlake Left|+1 Millican 126kts Goering|Timing KBDN pattern|30 taxi|2:00"}, { name: "2016 HWD Rally Leg 1", route: "KHWD|Start Taxi|6:00 Runup|0:30 Taxi|1:00 Takeoff Climb|5500|4:35 VPDUB Climb|5500|3:25 65CN 3CN7 57CN Left|+1 2CL1 Left|+1 O37 Left|+1 JELLYSFERRY HOWIE|Timing KRDD Pattern|0:45 Taxi|2:00"}, { name: "2016 HWD Rally Leg 2", route: "KRDD|Start Taxi|6:00 Runup|0:30 Taxi|2:00 Takeoff Climb|8500|1:20 REDDING.POND Climb|8500|7:40 O89 KAAT Left 1Q2 KSVE Left H37 Left CONSTANTIA2 HALLELUJAH2|Timing KRTS pattern|30 taxi|2:00"}, { name: "2017 HWD Rally Leg 1", route: "KHWD|Start Taxi|6:00 Runup|0:30 Taxi|1:00 Takeoff Climb|4500|8:00 10@6 9CL9 63CL THUNDERHILL 60@10 90CL 0O4 350@10 126kts JELLYSFERRY 126kts HOWIE|Timing KRDD Pattern|0:45 Taxi|4:00" }, { name: "2017 HWD Rally Leg 2", route: "KRDD|Start Taxi|6:00 Runup|0:30 Taxi|2:00 Takeoff Climb|8500|6:40 CASCADEHIGHWAY Climb|8500|8:20 KLKV 210@2 EAGLEVILLE DUCKLAKEPASS 210@5 O39 209@8 AHC 148@6 126kts CONSTANTIA2 126kts Hallelujah2|Timing KRTS Pattern|0:30 Taxi|2:00" }, { name: "San Jose to San Diego", route: "KSJC LICKE SNS V25 REDIN KMYF" }, { name: "San Diego to San Jose", route: "KMYF JOPDO CARIF V23 LAX V299 VTU V25 PRB SARDO RANCK V485 LICKE KSJC" }, { name: "San Jose to Renton", route: "KSJC SUNOL RBL V23 SEA KRNT" }, { name: "Renton to Willows", route: "KRNT SEA V495 BTG V23 RBL KWLW" }, { name: "SJC to SEA #1", route: "ksjc sunol 135kts 300@12 rdd 0kts pdx ksea" }, { name: "SJC to SEA #2", route: "KSJC SJC V334 SAC V23 BTG V495 SEA KSEA" }, { name: "SJC to SEA #3", route: "KSJC SJC V334 SAC V23 FJS V495 SEA KSEA" }, { name: "Roseburg to San Jose", route: "KRBG RBG KOLER V495 FJS V23 RBL SUNOL KSJC" }, { name: "SAN to DEN", route: "KSAN HAILE V514 LYNSY V8 JNC V134 BINBE KDEN" }, { name: "Denver to Minneapolis", route: "KDEN DVV V8 AKO V80 FSD V148 RWF V412 FCM KMSP" }, { name: "Reno to Chicago", route: "KRNO FMG V6 LLC V32 CEVAR V200 STACO V32 FBR V6 MBW V100 RFD V171 SIMMN V172 DPA KORD" }, { name: "Denver to Oklahoma City", route: "KDEN AVNEW V366 HGO V263 LAA V304 LBL V507 MMB V17 IRW KOKC" }, { name: "Stockton to Hawthorne 1", route: "KSCK MOD V113 PXN V107 AVE V137 GMN V23 LAX KHHR" }, { name: "Stockton to Hawthorne 2", route: "ksck patyy v113 rom v485 exert v25 lax khhr" }, { name: "Long View to Corpus Christi", route: "KGGG PIPES V289 LFK V13 WORRY KCRP" }, { name: "Austin Exec to Beaumont 1", route: "KEDC HOOKK V306 TNV V574 IAH V222 SHINA KBPT" }, { name: "Austin Exec to Beaumont 2", route: "KEDC HOOKK V306 DAS KBPT" }, { name: "Austin Exec to Beaumont 3", route: "KEDC HOOKK V306 TNV DAS KBPT" }, { name: "Savannah to Daytona Beach", route: "KSAV KELER V437 COKES KDAB" }, { name: "Philly to Hartford 1", route: "KPNE ARD V276 DIXIE V16 JFK V229 SNIVL KHFD" }, { name: "Philly to Hartford 2", route: "KPNE ARD V276 MANTA V139 RICED MAD KHFD" }, { name: "Philly to Hartford 3", route: "KPNE DITCH V312 DRIFT V139 SARDI V308 ORW KHFD" }, { name: "Philly to Hartford 4", route: "KPNE ZIDET V479 ARD V433 LGA V99 YALER KHFD" }, { name: "West Georgie Regional to Frankfort, KY 1", route: "kctj noone nello v5 gqo kfft" }, { name: "West Georgie Regional to Frankfort, KY 2", route: "kctj felto v243 gqo v333 hyk v512 clegg kfft" }, { name: "West Georgie Regional to Frankfort, KY 3", route: "kctj rmg v333 hyk v512 clegg kfft" }, { name: "West Georgie Regional to Frankfort, KY 4", route: "kctj rmg v333 hch v51 lvt v493 hyk kfft" }, { name: "Raleigh / Durham to Baltimore Martin State 1", route: "KRDU aimhi KMTN" }, { name: "Raleigh / Durham to Baltimore Martin State 2", route: "KRDU rdu v155 lvl ric ott KMTN" }, { name: "Raleigh / Durham to Baltimore Martin State 3", route: "KRDU rdu v155 mange v157 colin v33 ott v433 paleo KMTN" }, { name: "Raleigh / Durham to Baltimore Martin State 4", route: "KRDU rdu v155 LVL V157 RIC V16 PXT V93 GRACO KMTN" }, { name: "Roswell, NM to Longview, TX", route: "KROW Climb|9500|6:00 070@14 HOB V68 PIZON 050@16 V16 ABI V62 JEN V94 OTTIF KGGG" }, { name: "Lubbock to Longview 1", route: "KLBB JEN CQY KGGG" }, { name: "Lubbock to Longview 2", route: "KLBB ralls v102 gth v278 byp v114 awlar KGGG" }, { name: "Lubbock to Longview 3", route: "KLBB hydro v62 jen v94 ottif KGGG" }, { name: "Stockton, CA to North Las Vegas 1", route: "KSCK ECA V244 OAL V105 LUCKY KVGT" }, { name: "Stockton, CA to North Las Vegas 2", route: "KSCK ehf v197 pmd v12 basal v394 oasys KVGT" }, { name: "Bakersfield to Santa Rosa 1", route: "KBFL EHF V248 AVE OAK SAU KSTS" }, { name: "Bakersfield to Santa Rosa 2", route: "KBFL EHF V248 AVE PXN V301 SUNOL KSTS" }, { name: "Bakersfield to Santa Rosa 3", route: "KBFL SCRAP V248 AVE V107 OAK V195 CROIT V108 STS KSTS" }, { name: "Bakersfield to Santa Rosa 4", route: "KBFL EHF V23 SAC V494 SNUPY KSTS" }, { name: "Bakersfield to Santa Rosa 5", route: "KBFL EHF V248 AVE V137 SNS V230 SHOEY V27 PYE KSTS" }, { name: "Bakersfield to Santa Rosa 6", route: "KBFL EHF V23 CZQ MOD OAKEY KSTS" }, { name: "Bakersfield to Santa Rosa 7", route: "KBFL EHF V23 LIN CCR CROIT SGD KSTS" }, { name: "Bakersfield to Santa Rosa 8", route: "KBFL EHF V248 AVE V107 PXN SGD KSTS" }, { name: "Great Falls to Boeing King Field 1", route: "KGTF GTF V120 MLP V2 GEG V120 NORMY KBFI" }, { name: "Great Falls to Boeing King Field 2", route: "KGTF GTF V120 MLP J70 SEA KBFI" }, { name: "Great Falls to Boeing King Field 3", route: "KGTF GTF V187 ELN V2 SEA KBFI" }, { name: "Great Falls to Boeing King Field 4", route: "KGTF KEETA Q144 ZIRAN KBFI" }, { name: "Boise to Centenial 1", route: "KBOI BOI V4 LAR RAMMS NIWOT KAPA" }, { name: "Boise to Centenial 2", route: "KBOI ROARR PIH J20 OCS J154 AVVVS KAPA" }, { name: "Boise to Centenial 3", route: "KBOI CANEK V4 OCS V328 DOBEE V356 ELORE KAPA" }, { name: "Boise to Centenial 4", route: "KBOI BOI J54 PIH J20 OCS J52 FQF KAPA" }, { name: "St. Louis to Birmingham 1", route: "KSTL SPUDZ V125 DUEAS V540 CNG V67 SYI V321 BOAZE V115 COLIG KBHM" }, { name: "St. Louis to Birmingham 2", route: "KSTL ODUJY FAM CGI JKS MSL VUZ KBHM" }, { name: "Chattanoga to Augusta 1", route: "KCHA ODF AHN V417 MSTRS KAGS" }, { name: "Chattanoga to Augusta 2", route: "KCHA GQO NELLO CCATT T292 JACET KAGS" }, { name: "Chattanoga to Augusta 3", route: "KCHA GQO ATL ANNAN KAGS" }, { name: "Chattanoga to Augusta 4", route: "KCHA HOCHE V5 AHN V417 IRQ KAGS" }, { name: "Concord, NC to Richmond, VA 1", route: "KJQF SBV V20 RIC KRIC" }, { name: "Concord, NC to Richmond, VA 2", route: "KJQF GIZMO V143 GSO V266 SBV V20 RIC KRIC" }, { name: "Concord, NC to Richmond, VA 3", route: "KJQF GSO SBV V20 RIC KRIC" }, { name: "Concord, NC to Richmond, VA 4", route: "KJQF GIZMO V454 LVL V157 RIC KRIC" }, { name: "Buffalo, NY to Portland, ME 1", route: "KBUF BUF V2 UCA V496 NEETS V39 LIMER KPWM" }, { name: "Buffalo, NY to Portland, ME 2", route: "KBUF HANKK AUDIL PUPPY GFL KPWM" }, { name: "Buffalo, NY to Portland, ME 3", route: "KBUF BUF V14 GGT V428 UCA V496 ENE KPWM" }, { name: "Buffalo, NY to Portland, ME 4", route: "KBUF JOSSY Q935 PONCT ARIME CDOGG ENE KPWM" }, { name: "Moline, IL to Battle Creek, MI 1", route: "KMLI GENSO V8 NOMES V156 AZO KBTL" }, { name: "Moline, IL to Battle Creek, MI 2", route: "KMLI OBK J547 PMM KBTL" }, { name: "Moline, IL to Battle Creek, MI 3", route: "KMLI PLL OBK ELX KBTL" }, { name: "Moline, IL to Battle Creek, MI 4", route: "KMLI PLL RFD V100 ELX AZO KBTL" }, { name: "Green Bay, WI to Indianapolis, IN 1", route: "KGRB OKK V305 WELDO KIND" }, { name: "Green Bay, WI to Indianapolis, IN 2", route: "KGRB GRB J101 BAE J89 OBK EON V24 VHP KIND" }, { name: "Green Bay, WI to Indianapolis, IN 3", route: "KGRB WAFLE V7 BVT V399 ADVAY KIND" }, { name: "Green Bay, WI to Indianapolis, IN 4", route: "KGRB WAFLE V7 BVT VHP KIND" }, { name: "Anoka County, MN to Springfield, IL 1", route: "KANE KANAC V97 ODI V129 GROWL KSPI" }, { name: "Anoka County, MN to Springfield, IL 2", route: "KANE GEP PRIOR FGT V411 RST V67 ULAXY KSPI" }, { name: "Anoka County, MN to Springfield, IL 3", route: "KANE GEP PRIOR FGT V411 RST V503 CID V67 ULAXY KSPI" }, { name: "Anoka County, MN to Springfield, IL 4", route: "KANE WAGNR V510 ODI V129 SPI KSPI" }, { name: "Little Rock to Souix City 1", route: "KLIT MCI J41 OVR KSUX" }, { name: "Little Rock to Souix City 2", route: "KLIT ROLAN V534 HAAWK V71 SGF V159 SUX KSUX" }, { name: "Little Rock to Souix City 3 ", route: "KLIT ROLAN V534 SCRAN V527 RZC V13 BUM V71 PANNY KSUX" }, { name: "Little Rock to Souix City 4", route: "KLIT ROLAN V534 SCRAN V527 RZC V13 EOS V307 CNU V131 TOP PWE SUX KSUX" } ]; setupUserWaypoints(); print("let expectedFlightPlans = ["); for (let i = 0; i < flightPlans.length; ++i) { let flightPlanName = flightPlans[i].name; let flightPlanRoute = flightPlans[i].route; let flightPlan = new FlightPlan(flightPlanName, flightPlanRoute); flightPlan.parseRoute(); flightPlan.calculate(); let totalTime = flightPlan.totalTime(); let gateTime = flightPlan.gateTime(); let expectedGateTimeString; if (Math.abs(Time.differenceBetween(totalTime, gateTime).seconds()) == 0) expectedGateTimeString = "undefined"; else expectedGateTimeString = "new Time(\"" + gateTime + "\")"; print(" new ExpectedFlightPlan(\"" + flightPlanName + "\", \"" + flightPlanRoute + "\", \"" + flightPlan.resolvedRoute() + "\", new Time(\"" + totalTime + "\"), " + expectedGateTimeString + "),"); } print("];"); print("# Created " + flightPlans.length + " flight plans"); } // createTestRoutes();