Skip to content

Commit

Permalink
Merge remote-tracking branch 'entur/otp2_sorlandsbanen' into otp2_ent…
Browse files Browse the repository at this point in the history
…ur_develop
  • Loading branch information
t2gran committed Oct 21, 2024
2 parents fb1a836 + e416a17 commit 219b43d
Show file tree
Hide file tree
Showing 32 changed files with 718 additions and 345 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,79 @@
package org.opentripplanner.ext.sorlandsbanen;

import org.opentripplanner.raptor.api.model.RaptorAccessEgress;
import org.opentripplanner.raptor.api.model.RaptorTransferConstraint;
import org.opentripplanner.raptor.spi.RaptorCostCalculator;
import org.opentripplanner.routing.algorithm.raptoradapter.transit.TripSchedule;
import org.opentripplanner.routing.algorithm.raptoradapter.transit.cost.RaptorCostConverter;
import org.opentripplanner.transit.model.basic.TransitMode;


/**
* This cost calculator increases the cost on mode coach by adding an extra reluctance. The
* reluctance is hardcoded in this class and cannot be configured.
*/
class CoachCostCalculator<T extends TripSchedule> implements RaptorCostCalculator<T> {

private static final int EXTRA_RELUCTANCE_ON_COACH = RaptorCostConverter.toRaptorCost(0.6);

private final RaptorCostCalculator<T> delegate;

CoachCostCalculator(RaptorCostCalculator<T> delegate) {
this.delegate = delegate;
}

@Override
public int boardingCost(
boolean firstBoarding,
int prevArrivalTime,
int boardStop,
int boardTime,
T trip,
RaptorTransferConstraint transferConstraints
) {
return delegate.boardingCost(
firstBoarding,
prevArrivalTime,
boardStop,
boardTime,
trip,
transferConstraints
);
}

@Override
public int onTripRelativeRidingCost(int boardTime, T tripScheduledBoarded) {
return delegate.onTripRelativeRidingCost(boardTime, tripScheduledBoarded);
}

@Override
public int transitArrivalCost(
int boardCost,
int alightSlack,
int transitTime,
T trip,
int toStop
) {
int cost = delegate.transitArrivalCost(boardCost, alightSlack, transitTime, trip, toStop);
if(trip.transitReluctanceFactorIndex() == TransitMode.COACH.ordinal()) {
cost += transitTime * EXTRA_RELUCTANCE_ON_COACH;
}
return cost;
}

@Override
public int waitCost(int waitTimeInSeconds) {
return delegate.waitCost(waitTimeInSeconds);
}

@Override
public int calculateRemainingMinCost(int minTravelTime, int minNumTransfers, int fromStop) {
return delegate.calculateRemainingMinCost(minTravelTime, minNumTransfers, fromStop);
}

@Override
public int costEgress(RaptorAccessEgress egress) {
return delegate.costEgress(egress);
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,103 @@
package org.opentripplanner.ext.sorlandsbanen;

import java.util.Collection;
import java.util.function.BiFunction;
import org.opentripplanner.framework.geometry.WgsCoordinate;
import org.opentripplanner.model.GenericLocation;
import org.opentripplanner.raptor.api.path.RaptorPath;
import org.opentripplanner.raptor.spi.ExtraMcRouterSearch;
import org.opentripplanner.raptor.spi.RaptorTransitDataProvider;
import org.opentripplanner.routing.algorithm.raptoradapter.router.street.AccessEgresses;
import org.opentripplanner.routing.algorithm.raptoradapter.transit.RoutingAccessEgress;
import org.opentripplanner.routing.algorithm.raptoradapter.transit.TransitLayer;
import org.opentripplanner.routing.algorithm.raptoradapter.transit.TripSchedule;
import org.opentripplanner.routing.algorithm.raptoradapter.transit.request.RaptorRoutingRequestTransitData;
import org.opentripplanner.routing.api.request.RouteRequest;
import org.opentripplanner.transit.model.framework.FeedScopedId;
import org.opentripplanner.transit.model.site.StopLocation;

/**
* This is basically a big hack to produce results containing "Sørlandsbanen" in Norway. This
* railroad line is slow and goes inland fare from where people live. Despite this, people and the
* operator want to show it in the results for log travel along the southern part of Norway where
* ii is an option. Tuning the search has proven to be challenging. It is solved here by doing
* two searches. One normal search and one where the rail is given a big cost advantage over coach.
* If train results are found in the second search, then it is added to the results of the first
* search. Everything found in the first search is always returned.
*/
public class EnturSorlandsbanenService {

private static final double SOUTH_BOARDER_LIMIT = 59.1;
private static final int MIN_DISTANCE_LIMIT = 120_000;


public ExtraMcRouterSearch<TripSchedule> createMcRouterFactory(RouteRequest request, AccessEgresses accessEgresses, TransitLayer transitLayer) {
WgsCoordinate from = findStopCoordinate(
request.from(),
accessEgresses.getAccesses(),
transitLayer
);
WgsCoordinate to = findStopCoordinate(request.to(), accessEgresses.getEgresses(), transitLayer);

if (from.latitude() > SOUTH_BOARDER_LIMIT && to.latitude() > SOUTH_BOARDER_LIMIT) {
return null;
}

double distance = from.distanceTo(to);
if (distance < MIN_DISTANCE_LIMIT) {
return null;
}

return new ExtraMcRouterSearch<>() {
@Override
public RaptorTransitDataProvider<TripSchedule> createTransitDataAlternativeSearch(RaptorTransitDataProvider<TripSchedule> transitDataMainSearch) {
return new RaptorRoutingRequestTransitData(
(RaptorRoutingRequestTransitData)transitDataMainSearch,
new CoachCostCalculator<>(transitDataMainSearch.multiCriteriaCostCalculator())
);
}

@Override
public BiFunction<Collection<RaptorPath<TripSchedule>>, Collection<RaptorPath<TripSchedule>>, Collection<RaptorPath<TripSchedule>>> merger() {
return new MergePaths<>();
}
};
}

/**
* Find a coordinate matching the given location, in order:
* - First return the coordinate of the location if it exists.
* - Then loop through the access/egress stops and try to find the
* stop or station given by the location id, return the stop/station coordinate.
* - Return the fist stop in the access/egress list coordinate.
*/
@SuppressWarnings("ConstantConditions")
private static WgsCoordinate findStopCoordinate(
GenericLocation location,
Collection<? extends RoutingAccessEgress> accessEgress,
TransitLayer transitLayer
) {
if (location.lat != null) {
return new WgsCoordinate(location.lat, location.lng);
}

StopLocation firstStop = null;
for (RoutingAccessEgress it : accessEgress) {
StopLocation stop = transitLayer.getStopByIndex(it.stop());
if (stop.getId().equals(location.stopId)) {
return stop.getCoordinate();
}
if (idIsParentStation(stop, location.stopId)) {
return stop.getParentStation().getCoordinate();
}
if (firstStop == null) {
firstStop = stop;
}
}
return firstStop.getCoordinate();
}

private static boolean idIsParentStation(StopLocation stop, FeedScopedId pId) {
return stop.getParentStation() != null && stop.getParentStation().getId().equals(pId);
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
package org.opentripplanner.ext.sorlandsbanen;

import java.util.Collection;
import java.util.HashMap;
import java.util.Map;
import java.util.function.BiFunction;
import org.opentripplanner.raptor.api.model.RaptorTripSchedule;
import org.opentripplanner.raptor.api.path.PathLeg;
import org.opentripplanner.raptor.api.path.RaptorPath;
import org.opentripplanner.routing.algorithm.raptoradapter.transit.request.TripScheduleWithOffset;
import org.opentripplanner.transit.model.basic.TransitMode;

/**
* Strategy for merging the main results and the extra rail results from Sorlandsbanen.
* Everything from the main result is kept, and any additional rail results from the alternative
* search are added.
*/
class MergePaths<T extends RaptorTripSchedule> implements BiFunction<Collection<RaptorPath<T>>, Collection<RaptorPath<T>>, Collection<RaptorPath<T>>> {

@Override
public Collection<RaptorPath<T>> apply(Collection<RaptorPath<T>> main, Collection<RaptorPath<T>> railAlternatives) {
Map<PathKey, RaptorPath<T>> result = new HashMap<>();
addAllToMap(result, main);
addRailToMap(result, railAlternatives);
return result.values();
}

private void addAllToMap(Map<PathKey, RaptorPath<T>> map, Collection<RaptorPath<T>> paths) {
for (var it : paths) {
map.put(new PathKey(it), it);
}
}

private void addRailToMap(Map<PathKey, RaptorPath<T>> map, Collection<RaptorPath<T>> paths) {
for (var it : paths) {
if (hasRail(it)) {
map.put(new PathKey(it), it);
}
}
}

private static boolean hasRail(RaptorPath<?> path) {
return path
.legStream()
.filter(PathLeg::isTransitLeg)
.anyMatch(leg -> {
var trip = (TripScheduleWithOffset) leg.asTransitLeg().trip();
var mode = trip.getOriginalTripPattern().getMode();
return mode == TransitMode.RAIL;
});
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
package org.opentripplanner.ext.sorlandsbanen;

import org.opentripplanner.raptor.api.path.PathLeg;
import org.opentripplanner.raptor.api.path.RaptorPath;


/**
* Uses a hash to create a key for access, egress and transit legs in a path. Transfers
* are not included. The key is used to exclude duplicates. This approach may drop valid results
* when there is a hash collision, but this whole sandbox feature is a hack - so we can tolerate
* this here.
*/
final class PathKey {

private final int hash;

PathKey(RaptorPath<?> path) {
this.hash = hash(path);
}

private static int hash(RaptorPath<?> path) {
if (path == null) {
return 0;
}
int result = 1;

PathLeg<?> leg = path.accessLeg();

while (!leg.isEgressLeg()) {
result = 31 * result + leg.toStop();
result = 31 * result + leg.toTime();

if (leg.isTransitLeg()) {
result = 31 * result + leg.asTransitLeg().trip().pattern().debugInfo().hashCode();
}
leg = leg.nextLeg();
}
result = 31 * result + leg.toTime();

return result;
}

@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (o.getClass() != PathKey.class) {
return false;
}
return hash == ((PathKey) o).hash;
}

@Override
public int hashCode() {
return hash;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
package org.opentripplanner.ext.sorlandsbanen.configure;

import dagger.Module;
import dagger.Provides;
import javax.annotation.Nullable;
import org.opentripplanner.ext.sorlandsbanen.EnturSorlandsbanenService;
import org.opentripplanner.framework.application.OTPFeature;

@Module
public class EnturSorlandsbanenModule {

@Provides
@Nullable
EnturSorlandsbanenService providesEnturSorlandsbanenService() {
return OTPFeature.Sorlandsbanen.isOn() ? new EnturSorlandsbanenService() : null;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -118,6 +118,11 @@ public enum OTPFeature {
SandboxAPIGeocoder(false, true, "Enable the Geocoder API."),
SandboxAPIMapboxVectorTilesApi(false, true, "Enable Mapbox vector tiles API."),
SandboxAPIParkAndRideApi(false, true, "Enable park-and-ride endpoint."),
Sorlandsbanen(
false,
true,
"Include train Sørlandsbanen in results when searchig in south of Norway. Only relevant in Norway."
),
TransferAnalyzer(false, true, "Analyze transfers during graph build.");

private static final Object TEST_LOCK = new Object();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -163,6 +163,20 @@ public WgsCoordinate roundToApproximate100m() {
return new WgsCoordinate(lat, lng);
}

/**
* Compute a farly accurate distance between two coordinates. Use the fast version in
* {@link SphericalDistanceLibrary} if many computations are needed. Return the distance in
* meters between the two coordinates.
*/
public double distanceTo(WgsCoordinate other) {
return SphericalDistanceLibrary.distance(
this.latitude,
this.longitude,
other.latitude,
other.longitude
);
}

/**
* Return a new coordinate that is moved an approximate number of meters east.
*/
Expand Down
Original file line number Diff line number Diff line change
@@ -1,13 +1,15 @@
package org.opentripplanner.raptor;

import java.util.stream.Collectors;
import javax.annotation.Nullable;
import org.opentripplanner.raptor.api.model.RaptorTripSchedule;
import org.opentripplanner.raptor.api.request.RaptorRequest;
import org.opentripplanner.raptor.api.response.RaptorResponse;
import org.opentripplanner.raptor.configure.RaptorConfig;
import org.opentripplanner.raptor.service.DefaultStopArrivals;
import org.opentripplanner.raptor.service.HeuristicSearchTask;
import org.opentripplanner.raptor.service.RangeRaptorDynamicSearch;
import org.opentripplanner.raptor.spi.ExtraMcRouterSearch;
import org.opentripplanner.raptor.spi.RaptorTransitDataProvider;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
Expand All @@ -23,8 +25,16 @@ public class RaptorService<T extends RaptorTripSchedule> {

private final RaptorConfig<T> config;

public RaptorService(RaptorConfig<T> config) {
@Nullable
private final ExtraMcRouterSearch<T> extraMcSearch;

public RaptorService(RaptorConfig<T> config, @Nullable ExtraMcRouterSearch<T> extraMcSearch) {
this.config = config;
this.extraMcSearch = extraMcSearch;
}

public RaptorService(RaptorConfig<T> config) {
this(config, null);
}

public RaptorResponse<T> route(
Expand All @@ -35,7 +45,8 @@ public RaptorResponse<T> route(
RaptorResponse<T> response;

if (request.isDynamicSearch()) {
response = new RangeRaptorDynamicSearch<>(config, transitData, request).route();
response =
new RangeRaptorDynamicSearch<>(config, transitData, extraMcSearch, request).route();
} else {
response = routeUsingStdWorker(transitData, request);
}
Expand Down
Loading

0 comments on commit 219b43d

Please sign in to comment.