|
2 | 2 |
|
3 | 3 | import static org.junit.jupiter.api.Assertions.assertEquals; |
4 | 4 | import static org.opentripplanner.updater.spi.UpdateResultAssertions.assertFailure; |
| 5 | +import static org.opentripplanner.updater.spi.UpdateResultAssertions.assertSuccess; |
5 | 6 |
|
6 | 7 | import org.junit.jupiter.api.Test; |
7 | 8 | import org.opentripplanner.transit.model._data.TransitTestEnvironment; |
8 | 9 | import org.opentripplanner.transit.model._data.TransitTestEnvironmentBuilder; |
9 | 10 | import org.opentripplanner.transit.model._data.TripInput; |
| 11 | +import org.opentripplanner.transit.model.basic.TransitMode; |
10 | 12 | import org.opentripplanner.transit.model.site.RegularStop; |
11 | 13 | import org.opentripplanner.updater.spi.UpdateError; |
12 | 14 | import org.opentripplanner.updater.trip.RealtimeTestConstants; |
13 | 15 | import org.opentripplanner.updater.trip.SiriTestHelper; |
| 16 | +import uk.org.siri.siri21.VehicleModesEnumeration; |
14 | 17 |
|
15 | 18 | class FuzzyTripMatchingTest implements RealtimeTestConstants { |
16 | 19 |
|
@@ -41,8 +44,11 @@ void testUpdateJourneyWithFuzzyMatching() { |
41 | 44 | ) |
42 | 45 | .buildEstimatedTimetableDeliveries(); |
43 | 46 | var result = siri.applyEstimatedTimetableWithFuzzyMatcher(updates); |
44 | | - assertEquals(1, result.successful()); |
45 | | - assertTripUpdated(env); |
| 47 | + assertSuccess(result); |
| 48 | + assertEquals( |
| 49 | + "UPDATED | A 0:00:15 0:00:15 | B 0:00:25 0:00:25", |
| 50 | + env.tripData(TRIP_1_ID).showTimetable() |
| 51 | + ); |
46 | 52 | } |
47 | 53 |
|
48 | 54 | /** |
@@ -73,10 +79,92 @@ void testUpdateJourneyWithFuzzyMatchingAndMissingAimedDepartureTime() { |
73 | 79 | assertFailure(UpdateError.UpdateErrorType.NO_FUZZY_TRIP_MATCH, result); |
74 | 80 | } |
75 | 81 |
|
76 | | - private static void assertTripUpdated(TransitTestEnvironment env) { |
| 82 | + /** |
| 83 | + * Two RAIL trips with identical stops and times but different internalPlanningCodes. |
| 84 | + * The SIRI update has a non-matching VehicleJourneyRef but includes a VehicleRef that |
| 85 | + * corresponds to one trip's planning code. The matcher should disambiguate using VehicleRef. |
| 86 | + */ |
| 87 | + @Test |
| 88 | + void testFuzzyMatchByVehicleRefForRailTrip() { |
| 89 | + var railRoute = ENV_BUILDER.route("RailRoute", r -> r.withMode(TransitMode.RAIL)); |
| 90 | + |
| 91 | + var railTrip1 = TripInput.of("RailTrip1") |
| 92 | + .withRoute(railRoute) |
| 93 | + .withNetexInternalPlanningCode("47") |
| 94 | + .addStop(STOP_A, "0:00:10", "0:00:11") |
| 95 | + .addStop(STOP_B, "0:00:20", "0:00:21"); |
| 96 | + |
| 97 | + var railTrip2 = TripInput.of("RailTrip2") |
| 98 | + .withRoute(railRoute) |
| 99 | + .withNetexInternalPlanningCode("48") |
| 100 | + .addStop(STOP_A, "0:00:10", "0:00:11") |
| 101 | + .addStop(STOP_B, "0:00:20", "0:00:21"); |
| 102 | + |
| 103 | + var env = ENV_BUILDER.addTrip(railTrip1).addTrip(railTrip2).build(); |
| 104 | + |
| 105 | + var siri = SiriTestHelper.of(env); |
| 106 | + |
| 107 | + var updates = siri |
| 108 | + .etBuilder() |
| 109 | + .withFramedVehicleJourneyRef(builder -> |
| 110 | + builder.withServiceDate(env.defaultServiceDate()).withVehicleJourneyRef("NONEXISTENT") |
| 111 | + ) |
| 112 | + .withVehicleRef("47") |
| 113 | + .withVehicleMode(VehicleModesEnumeration.RAIL) |
| 114 | + .withEstimatedCalls(builder -> |
| 115 | + builder |
| 116 | + .call(STOP_A) |
| 117 | + .departAimedExpected("00:00:11", "00:00:15") |
| 118 | + .call(STOP_B) |
| 119 | + .arriveAimedExpected("00:00:20", "00:00:25") |
| 120 | + ) |
| 121 | + .buildEstimatedTimetableDeliveries(); |
| 122 | + |
| 123 | + var result = siri.applyEstimatedTimetableWithFuzzyMatcher(updates); |
| 124 | + assertSuccess(result); |
77 | 125 | assertEquals( |
78 | 126 | "UPDATED | A 0:00:15 0:00:15 | B 0:00:25 0:00:25", |
79 | | - env.tripData(TRIP_1_ID).showTimetable() |
| 127 | + env.tripData("RailTrip1").showTimetable() |
| 128 | + ); |
| 129 | + } |
| 130 | + |
| 131 | + /** |
| 132 | + * The fuzzy matcher should still resolve the trip via VehicleRef → internalPlanningCode |
| 133 | + * when only DatedVehicleJourneyRef is provided. |
| 134 | + */ |
| 135 | + @Test |
| 136 | + void testFuzzyMatchByVehicleRefWithDatedVehicleJourneyRefOnly() { |
| 137 | + var railRoute = ENV_BUILDER.route("RailRoute2", r -> r.withMode(TransitMode.RAIL)); |
| 138 | + |
| 139 | + var railTrip = TripInput.of("RailTrip3") |
| 140 | + .withRoute(railRoute) |
| 141 | + .withNetexInternalPlanningCode("406") |
| 142 | + .addStop(STOP_A, "0:00:10", "0:00:11") |
| 143 | + .addStop(STOP_B, "0:00:20", "0:00:21"); |
| 144 | + |
| 145 | + var env = ENV_BUILDER.addTrip(railTrip).build(); |
| 146 | + |
| 147 | + var siri = SiriTestHelper.of(env); |
| 148 | + |
| 149 | + var updates = siri |
| 150 | + .etBuilder() |
| 151 | + .withDatedVehicleJourneyRef("406:2026-02-17") |
| 152 | + .withVehicleRef("406") |
| 153 | + .withVehicleMode(VehicleModesEnumeration.RAIL) |
| 154 | + .withEstimatedCalls(builder -> |
| 155 | + builder |
| 156 | + .call(STOP_A) |
| 157 | + .departAimedExpected("00:00:11", "00:00:15") |
| 158 | + .call(STOP_B) |
| 159 | + .arriveAimedExpected("00:00:20", "00:00:25") |
| 160 | + ) |
| 161 | + .buildEstimatedTimetableDeliveries(); |
| 162 | + |
| 163 | + var result = siri.applyEstimatedTimetableWithFuzzyMatcher(updates); |
| 164 | + assertSuccess(result); |
| 165 | + assertEquals( |
| 166 | + "UPDATED | A 0:00:15 0:00:15 | B 0:00:25 0:00:25", |
| 167 | + env.tripData("RailTrip3").showTimetable() |
80 | 168 | ); |
81 | 169 | } |
82 | 170 | } |
0 commit comments