Skip to content

Commit

Permalink
Merge remote-tracking branch 'upstream/dev-2.x' into netex-parking
Browse files Browse the repository at this point in the history
  • Loading branch information
leonardehrenfried committed Jul 17, 2024
2 parents cc659fa + 5917ddf commit 5813d2c
Show file tree
Hide file tree
Showing 15 changed files with 203 additions and 64 deletions.
1 change: 1 addition & 0 deletions docs/Changelog.md
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ based on merged pull requests. Search GitHub issues and pull requests for smalle
- Fix copy-on-write in TimetableSnapshot [#5941](https://github.com/opentripplanner/OpenTripPlanner/pull/5941)
- Generate documentation for OSM tag mappers [#5929](https://github.com/opentripplanner/OpenTripPlanner/pull/5929)
- Disable Legacy REST API by default [#5948](https://github.com/opentripplanner/OpenTripPlanner/pull/5948)
- Enforce non-null coordinates on multimodal station [#5971](https://github.com/opentripplanner/OpenTripPlanner/pull/5971)
[](AUTOMATIC_CHANGELOG_PLACEHOLDER_DO_NOT_REMOVE)

## 2.5.0 (2024-03-13)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@
import org.opentripplanner.transit.model.timetable.TripIdAndServiceDate;
import org.opentripplanner.transit.service.DefaultTransitService;
import org.opentripplanner.transit.service.StopModel;
import org.opentripplanner.transit.service.TransitEditorService;
import org.opentripplanner.transit.service.TransitModel;
import org.opentripplanner.updater.spi.UpdateError;
import uk.org.siri.siri20.VehicleModesEnumeration;
Expand Down Expand Up @@ -77,6 +78,7 @@ class AddedTripBuilderTest {

private final Deduplicator DEDUPLICATOR = new Deduplicator();
private final TransitModel TRANSIT_MODEL = new TransitModel(STOP_MODEL, DEDUPLICATOR);
private TransitEditorService transitService;
private EntityResolver ENTITY_RESOLVER;

@BeforeEach
Expand All @@ -101,6 +103,7 @@ void setUp() {

// Create transit model index
TRANSIT_MODEL.index();
transitService = new DefaultTransitService(TRANSIT_MODEL);

// Create the entity resolver only after the model has been indexed
ENTITY_RESOLVER =
Expand All @@ -110,7 +113,7 @@ void setUp() {
@Test
void testAddedTrip() {
var addedTrip = new AddedTripBuilder(
TRANSIT_MODEL,
transitService,
ENTITY_RESOLVER,
AbstractTransitEntity::getId,
TRIP_ID,
Expand Down Expand Up @@ -239,7 +242,7 @@ void testAddedTrip() {
@Test
void testAddedTripOnAddedRoute() {
var firstAddedTrip = new AddedTripBuilder(
TRANSIT_MODEL,
transitService,
ENTITY_RESOLVER,
AbstractTransitEntity::getId,
TRIP_ID,
Expand All @@ -265,7 +268,7 @@ void testAddedTripOnAddedRoute() {
var tripId2 = TransitModelForTest.id("TRIP_ID_2");

var secondAddedTrip = new AddedTripBuilder(
TRANSIT_MODEL,
transitService,
ENTITY_RESOLVER,
AbstractTransitEntity::getId,
tripId2,
Expand Down Expand Up @@ -316,7 +319,7 @@ void testAddedTripOnAddedRoute() {
@Test
void testAddedTripOnExistingRoute() {
var addedTrip = new AddedTripBuilder(
TRANSIT_MODEL,
transitService,
ENTITY_RESOLVER,
AbstractTransitEntity::getId,
TRIP_ID,
Expand Down Expand Up @@ -347,7 +350,7 @@ void testAddedTripOnExistingRoute() {
@Test
void testAddedTripWithoutReplacedRoute() {
var addedTrip = new AddedTripBuilder(
TRANSIT_MODEL,
transitService,
ENTITY_RESOLVER,
AbstractTransitEntity::getId,
TRIP_ID,
Expand Down Expand Up @@ -390,7 +393,7 @@ void testAddedTripWithoutReplacedRoute() {
@Test
void testAddedTripFailOnMissingServiceId() {
var addedTrip = new AddedTripBuilder(
TRANSIT_MODEL,
transitService,
ENTITY_RESOLVER,
AbstractTransitEntity::getId,
TRIP_ID,
Expand Down Expand Up @@ -445,7 +448,7 @@ void testAddedTripFailOnNonIncreasingDwellTime() {
);

var addedTrip = new AddedTripBuilder(
TRANSIT_MODEL,
transitService,
ENTITY_RESOLVER,
AbstractTransitEntity::getId,
TRIP_ID,
Expand Down Expand Up @@ -484,7 +487,7 @@ void testAddedTripFailOnTooFewCalls() {
.build()
);
var addedTrip = new AddedTripBuilder(
TRANSIT_MODEL,
transitService,
ENTITY_RESOLVER,
AbstractTransitEntity::getId,
TRIP_ID,
Expand Down Expand Up @@ -531,7 +534,7 @@ void testAddedTripFailOnUnknownStop() {
.build()
);
var addedTrip = new AddedTripBuilder(
TRANSIT_MODEL,
transitService,
ENTITY_RESOLVER,
AbstractTransitEntity::getId,
TRIP_ID,
Expand Down
46 changes: 21 additions & 25 deletions src/ext/java/org/opentripplanner/ext/siri/AddedTripBuilder.java
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,7 @@
import org.opentripplanner.transit.model.timetable.TripIdAndServiceDate;
import org.opentripplanner.transit.model.timetable.TripOnServiceDate;
import org.opentripplanner.transit.model.timetable.TripTimesFactory;
import org.opentripplanner.transit.service.TransitModel;
import org.opentripplanner.transit.service.TransitEditorService;
import org.opentripplanner.updater.spi.DataValidationExceptionMapper;
import org.opentripplanner.updater.spi.UpdateError;
import org.rutebanken.netex.model.BusSubmodeEnumeration;
Expand All @@ -52,7 +52,7 @@
class AddedTripBuilder {

private static final Logger LOG = LoggerFactory.getLogger(AddedTripBuilder.class);
private final TransitModel transitModel;
private final TransitEditorService transitService;
private final EntityResolver entityResolver;
private final ZoneId timeZone;
private final Function<Trip, FeedScopedId> getTripPatternId;
Expand All @@ -73,7 +73,7 @@ class AddedTripBuilder {

AddedTripBuilder(
EstimatedVehicleJourney estimatedVehicleJourney,
TransitModel transitModel,
TransitEditorService transitService,
EntityResolver entityResolver,
Function<Trip, FeedScopedId> getTripPatternId
) {
Expand Down Expand Up @@ -112,16 +112,16 @@ class AddedTripBuilder {

calls = CallWrapper.of(estimatedVehicleJourney);

this.transitModel = transitModel;
this.transitService = transitService;
this.entityResolver = entityResolver;
this.getTripPatternId = getTripPatternId;
timeZone = transitModel.getTimeZone();
timeZone = transitService.getTimeZone();

replacedTrips = getReplacedVehicleJourneys(estimatedVehicleJourney);
}

AddedTripBuilder(
TransitModel transitModel,
TransitEditorService transitService,
EntityResolver entityResolver,
Function<Trip, FeedScopedId> getTripPatternId,
FeedScopedId tripId,
Expand All @@ -139,9 +139,9 @@ class AddedTripBuilder {
String headsign,
List<TripOnServiceDate> replacedTrips
) {
this.transitModel = transitModel;
this.transitService = transitService;
this.entityResolver = entityResolver;
this.timeZone = transitModel.getTimeZone();
this.timeZone = transitService.getTimeZone();
this.getTripPatternId = getTripPatternId;
this.tripId = tripId;
this.operator = operator;
Expand All @@ -168,7 +168,7 @@ Result<TripUpdate, UpdateError> build() {
return UpdateError.result(tripId, NO_START_DATE);
}

FeedScopedId calServiceId = transitModel.getOrCreateServiceIdForDate(serviceDate);
FeedScopedId calServiceId = transitService.getOrCreateServiceIdForDate(serviceDate);
if (calServiceId == null) {
return UpdateError.result(tripId, NO_START_DATE);
}
Expand All @@ -181,7 +181,7 @@ Result<TripUpdate, UpdateError> build() {
}
route = createRoute(agency);
LOG.info("Adding route {} to transitModel.", route);
transitModel.getTransitModelIndex().addRoutes(route);
transitService.addRoutes(route);
}

Trip trip = createTrip(route, calServiceId);
Expand Down Expand Up @@ -221,14 +221,14 @@ Result<TripUpdate, UpdateError> build() {
RealTimeTripTimes tripTimes = TripTimesFactory.tripTimes(
trip,
aimedStopTimes,
transitModel.getDeduplicator()
transitService.getDeduplicator()
);
// validate the scheduled trip times
// they are in general superseded by real-time trip times
// but in case of trip cancellation, OTP will fall back to scheduled trip times
// therefore they must be valid
tripTimes.validateNonIncreasingTimes();
tripTimes.setServiceCode(transitModel.getServiceCodes().get(trip.getServiceId()));
tripTimes.setServiceCode(transitService.getServiceCodeForId(trip.getServiceId()));
pattern.add(tripTimes);
RealTimeTripTimes updatedTripTimes = tripTimes.copyScheduledTimes();

Expand Down Expand Up @@ -267,17 +267,14 @@ Result<TripUpdate, UpdateError> build() {

// Adding trip to index necessary to include values in graphql-queries
// TODO - SIRI: should more data be added to index?
transitModel.getTransitModelIndex().getTripForId().put(tripId, trip);
transitModel.getTransitModelIndex().getPatternForTrip().put(trip, pattern);
transitModel.getTransitModelIndex().getPatternsForRoute().put(route, pattern);
transitModel
.getTransitModelIndex()
.getTripOnServiceDateById()
.put(tripOnServiceDate.getId(), tripOnServiceDate);
transitModel
.getTransitModelIndex()
.getTripOnServiceDateForTripAndDay()
.put(new TripIdAndServiceDate(tripId, serviceDate), tripOnServiceDate);
transitService.addTripForId(tripId, trip);
transitService.addPatternForTrip(trip, pattern);
transitService.addPatternsForRoute(route, pattern);
transitService.addTripOnServiceDateById(tripOnServiceDate.getId(), tripOnServiceDate);
transitService.addTripOnServiceDateForTripAndDay(
new TripIdAndServiceDate(tripId, serviceDate),
tripOnServiceDate
);

return Result.success(new TripUpdate(stopPattern, updatedTripTimes, serviceDate));
}
Expand Down Expand Up @@ -312,8 +309,7 @@ private Route createRoute(Agency agency) {
*/
@Nullable
private Agency resolveAgency() {
return transitModel
.getTransitModelIndex()
return transitService
.getAllRoutes()
.stream()
.filter(r -> r != null && r.getOperator() != null && r.getOperator().equals(operator))
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -23,8 +23,8 @@
import org.opentripplanner.transit.model.timetable.Trip;
import org.opentripplanner.transit.model.timetable.TripTimes;
import org.opentripplanner.transit.service.DefaultTransitService;
import org.opentripplanner.transit.service.TransitEditorService;
import org.opentripplanner.transit.service.TransitModel;
import org.opentripplanner.transit.service.TransitService;
import org.opentripplanner.updater.TimetableSnapshotSourceParameters;
import org.opentripplanner.updater.spi.DataValidationExceptionMapper;
import org.opentripplanner.updater.spi.UpdateError;
Expand Down Expand Up @@ -56,9 +56,8 @@ public class SiriTimetableSnapshotSource implements TimetableSnapshotProvider {
* messages.
*/
private final SiriTripPatternCache tripPatternCache;
private final TransitModel transitModel;

private final TransitService transitService;
private final TransitEditorService transitService;

private final TimetableSnapshotManager snapshotManager;

Expand All @@ -72,7 +71,6 @@ public SiriTimetableSnapshotSource(
parameters,
() -> LocalDate.now(transitModel.getTimeZone())
);
this.transitModel = transitModel;
this.transitService = new DefaultTransitService(transitModel);
this.tripPatternCache =
new SiriTripPatternCache(tripPatternIdGenerator, transitService::getPatternForTrip);
Expand Down Expand Up @@ -115,7 +113,7 @@ public UpdateResult applyEstimatedTimetable(
var journeys = estimatedJourneyVersion.getEstimatedVehicleJourneies();
LOG.debug("Handling {} EstimatedVehicleJourneys.", journeys.size());
for (EstimatedVehicleJourney journey : journeys) {
results.add(apply(journey, transitModel, fuzzyTripMatcher, entityResolver));
results.add(apply(journey, transitService, fuzzyTripMatcher, entityResolver));
}
}
}
Expand All @@ -135,7 +133,7 @@ public TimetableSnapshot getTimetableSnapshot() {

private Result<UpdateSuccess, UpdateError> apply(
EstimatedVehicleJourney journey,
TransitModel transitModel,
TransitEditorService transitService,
@Nullable SiriFuzzyTripMatcher fuzzyTripMatcher,
EntityResolver entityResolver
) {
Expand All @@ -147,7 +145,7 @@ private Result<UpdateSuccess, UpdateError> apply(
result =
new AddedTripBuilder(
journey,
transitModel,
transitService,
entityResolver,
tripPatternIdGenerator::generateUniqueTripPatternId
)
Expand Down Expand Up @@ -265,7 +263,7 @@ private Result<TripUpdate, UpdateError> handleModifiedTrip(
pattern,
estimatedVehicleJourney,
serviceDate,
transitModel.getTimeZone(),
transitService.getTimeZone(),
entityResolver
)
.build();
Expand Down
36 changes: 26 additions & 10 deletions src/main/java/org/opentripplanner/model/TimetableSnapshot.java
Original file line number Diff line number Diff line change
Expand Up @@ -93,15 +93,15 @@ public class TimetableSnapshot {
* The compound key approach better reflects the fact that there should be only one Timetable per
* TripPattern and date.
*/
private Map<TripPattern, SortedSet<Timetable>> timetables = new HashMap<>();
private final Map<TripPattern, SortedSet<Timetable>> timetables;

/**
* For cases where the trip pattern (sequence of stops visited) has been changed by a realtime
* update, a Map associating the updated trip pattern with a compound key of the feed-scoped
* trip ID and the service date.
* TODO RT_AB: clarify if this is an index or the original source of truth.
*/
private Map<TripIdAndServiceDate, TripPattern> realtimeAddedTripPattern = new HashMap<>();
private final Map<TripIdAndServiceDate, TripPattern> realtimeAddedTripPattern;

/**
* This is an index of TripPatterns, not the primary collection. It tracks which TripPatterns
Expand All @@ -111,20 +111,36 @@ public class TimetableSnapshot {
* more than once.
* TODO RT_AB: More general handling of all realtime indexes outside primary data structures.
*/
private SetMultimap<StopLocation, TripPattern> patternsForStop = HashMultimap.create();
private final SetMultimap<StopLocation, TripPattern> patternsForStop;

/**
* Boolean value indicating that timetable snapshot is read only if true. Once it is true, it
* shouldn't be possible to change it to false anymore.
*/
private boolean readOnly = false;
private final boolean readOnly;

/**
* Boolean value indicating that this timetable snapshot contains changes compared to the state of
* the last commit if true.
*/
private boolean dirty = false;

public TimetableSnapshot() {
this(new HashMap<>(), new HashMap<>(), HashMultimap.create(), false);
}

private TimetableSnapshot(
Map<TripPattern, SortedSet<Timetable>> timetables,
Map<TripIdAndServiceDate, TripPattern> realtimeAddedTripPattern,
SetMultimap<StopLocation, TripPattern> patternsForStop,
boolean readOnly
) {
this.timetables = timetables;
this.realtimeAddedTripPattern = realtimeAddedTripPattern;
this.patternsForStop = patternsForStop;
this.readOnly = readOnly;
}

/**
* Returns an updated timetable for the specified pattern if one is available in this snapshot, or
* the originally scheduled timetable if there are no updates in this snapshot.
Expand Down Expand Up @@ -235,12 +251,15 @@ public TimetableSnapshot commit(TransitLayerUpdater transitLayerUpdater, boolean
throw new ConcurrentModificationException("This TimetableSnapshot is read-only.");
}

TimetableSnapshot ret = new TimetableSnapshot();
if (!force && !this.isDirty()) {
return null;
}
ret.timetables = Map.copyOf(timetables);
ret.realtimeAddedTripPattern = Map.copyOf(realtimeAddedTripPattern);
TimetableSnapshot ret = new TimetableSnapshot(
Map.copyOf(timetables),
Map.copyOf(realtimeAddedTripPattern),
ImmutableSetMultimap.copyOf(patternsForStop),
true
);

if (transitLayerUpdater != null) {
transitLayerUpdater.update(dirtyTimetables, timetables);
Expand All @@ -249,9 +268,6 @@ public TimetableSnapshot commit(TransitLayerUpdater transitLayerUpdater, boolean
this.dirtyTimetables.clear();
this.dirty = false;

ret.patternsForStop = ImmutableSetMultimap.copyOf(patternsForStop);

ret.readOnly = true; // mark the snapshot as henceforth immutable
return ret;
}

Expand Down
Loading

0 comments on commit 5813d2c

Please sign in to comment.