diff --git a/src/main/java/org/opentripplanner/middleware/triptracker/ManageTripTracking.java b/src/main/java/org/opentripplanner/middleware/triptracker/ManageTripTracking.java index 0e131ce75..1332bdf12 100644 --- a/src/main/java/org/opentripplanner/middleware/triptracker/ManageTripTracking.java +++ b/src/main/java/org/opentripplanner/middleware/triptracker/ManageTripTracking.java @@ -59,6 +59,7 @@ private static TrackingResponse doUpdateTracking(Request request, TripTrackingDa ); TripStatus tripStatus = TripStatus.getTripStatus(travelerPosition); trackedJourney.lastLocation().tripStatus = tripStatus; + trackedJourney.lastLocation().deviationMeters = travelerPosition.getDeviationMeters(); if (create) { Persistence.trackedJourneys.create(trackedJourney); diff --git a/src/main/java/org/opentripplanner/middleware/triptracker/TrackingLocation.java b/src/main/java/org/opentripplanner/middleware/triptracker/TrackingLocation.java index 087dc6694..25a755db9 100644 --- a/src/main/java/org/opentripplanner/middleware/triptracker/TrackingLocation.java +++ b/src/main/java/org/opentripplanner/middleware/triptracker/TrackingLocation.java @@ -18,8 +18,15 @@ public class TrackingLocation { public Date timestamp; + /** Deviation or on-time status computed for this location. */ public TripStatus tripStatus; + /** FIXME: Device location accuracy, reported by the device in a unit TBD. For reporting only. */ + public Double locationAccuracy; + + /** Perpendicular deviation, computed in meters, to the path closest to this location. */ + public Double deviationMeters; + public TrackingLocation() { // Needed for deserializing objects. } diff --git a/src/main/java/org/opentripplanner/middleware/triptracker/TravelerPosition.java b/src/main/java/org/opentripplanner/middleware/triptracker/TravelerPosition.java index 40bd3ae8e..104ee417c 100644 --- a/src/main/java/org/opentripplanner/middleware/triptracker/TravelerPosition.java +++ b/src/main/java/org/opentripplanner/middleware/triptracker/TravelerPosition.java @@ -13,6 +13,7 @@ import static org.opentripplanner.middleware.triptracker.ManageLegTraversal.getExpectedLeg; import static org.opentripplanner.middleware.triptracker.ManageLegTraversal.getNextLeg; import static org.opentripplanner.middleware.triptracker.ManageLegTraversal.getSegmentFromPosition; +import static org.opentripplanner.middleware.utils.GeometryUtils.getDistanceFromLine; public class TravelerPosition { @@ -81,4 +82,9 @@ public TravelerPosition(Leg nextLeg, Instant currentTime) { this.nextLeg = nextLeg; this.currentTime = currentTime; } + + /** Computes the current deviation in meters from the expected itinerary. */ + public double getDeviationMeters() { + return getDistanceFromLine(legSegmentFromPosition.start, legSegmentFromPosition.end, currentPosition); + } } diff --git a/src/main/java/org/opentripplanner/middleware/triptracker/TripStatus.java b/src/main/java/org/opentripplanner/middleware/triptracker/TripStatus.java index 8ff353a8d..e1cff6032 100644 --- a/src/main/java/org/opentripplanner/middleware/triptracker/TripStatus.java +++ b/src/main/java/org/opentripplanner/middleware/triptracker/TripStatus.java @@ -104,11 +104,7 @@ public static Instant getSegmentStartTime(TravelerPosition travelerPosition) { * Checks if the traveler's position is with an acceptable distance of the mode type. */ private static boolean isWithinModeRadius(TravelerPosition travelerPosition) { - double distanceFromExpected = getDistanceFromLine( - travelerPosition.legSegmentFromPosition.start, - travelerPosition.legSegmentFromPosition.end, - travelerPosition.currentPosition - ); + double distanceFromExpected = travelerPosition.getDeviationMeters(); double modeBoundary = getModeRadius(travelerPosition.legSegmentFromPosition.mode); return distanceFromExpected <= modeBoundary; } diff --git a/src/main/java/org/opentripplanner/middleware/utils/GeometryUtils.java b/src/main/java/org/opentripplanner/middleware/utils/GeometryUtils.java index 9f28bb1c9..d725fb6dd 100644 --- a/src/main/java/org/opentripplanner/middleware/utils/GeometryUtils.java +++ b/src/main/java/org/opentripplanner/middleware/utils/GeometryUtils.java @@ -22,7 +22,7 @@ public static double getDistance(Coordinates start, Coordinates end) { } /** - * Get the distance between a line and point. + * Get the distance in meters between a line and point. */ public static double getDistanceFromLine(Coordinates start, Coordinates end, Coordinates traveler) { double[] startXY = convertLatLonToXY(start.lat, start.lon); diff --git a/src/main/resources/latest-spark-swagger-output.yaml b/src/main/resources/latest-spark-swagger-output.yaml index 88ea9b539..a4bc8f939 100644 --- a/src/main/resources/latest-spark-swagger-output.yaml +++ b/src/main/resources/latest-spark-swagger-output.yaml @@ -2773,6 +2773,12 @@ definitions: - "ENDED" - "DEVIATED" - "COMPLETED" + locationAccuracy: + type: "number" + format: "double" + deviationMeters: + type: "number" + format: "double" TrackingResponse: type: "object" properties: diff --git a/src/test/java/org/opentripplanner/middleware/controllers/api/TrackedTripControllerTest.java b/src/test/java/org/opentripplanner/middleware/controllers/api/TrackedTripControllerTest.java index 30748a075..0d6d95bcc 100644 --- a/src/test/java/org/opentripplanner/middleware/controllers/api/TrackedTripControllerTest.java +++ b/src/test/java/org/opentripplanner/middleware/controllers/api/TrackedTripControllerTest.java @@ -262,6 +262,26 @@ void canGenerateInstructionAndStatus( assertEquals(status.name(), trackResponse.tripStatus); assertNotNull(trackResponse.journeyId); trackedJourney = Persistence.trackedJourneys.getById(trackResponse.journeyId); + + // Check that deviation fields get computed and recorded. + Double deviationMeters = trackedJourney.lastLocation().deviationMeters; + assertNotNull(deviationMeters); + assertNotEquals(0, deviationMeters); + + // Second request to update a journey + response = makeRequest( + TRACK_TRIP_PATH, + jsonPayload, + headers, + HttpMethod.POST + ); + + assertEquals(HttpStatus.OK_200, response.status); + trackResponse = JsonUtils.getPOJOFromJSON(response.responseBody, TrackingResponse.class); + assertNotEquals(0, trackResponse.frequencySeconds); + assertEquals(instruction, trackResponse.instruction, message); + assertNotNull(trackResponse.journeyId); + assertEquals(trackedJourney.id, trackResponse.journeyId); } private static Stream createInstructionAndStatusCases() {