Skip to content

Commit

Permalink
more detail in select link path reporting
Browse files Browse the repository at this point in the history
report route name in totals row
report proportions out of total iterations reaching destination
provide constituent rows below summary rows
  • Loading branch information
abyrd committed Jan 12, 2024
1 parent e5ae1c5 commit a4bf241
Show file tree
Hide file tree
Showing 2 changed files with 43 additions and 20 deletions.
56 changes: 39 additions & 17 deletions src/main/java/com/conveyal/r5/analyst/cluster/PathResult.java
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,6 @@

import com.conveyal.r5.analyst.StreetTimesAndModes;
import com.conveyal.r5.transit.TransitLayer;
import com.conveyal.r5.transit.TripPattern;
import com.conveyal.r5.transit.path.Path;
import com.conveyal.r5.transit.path.PatternSequence;
import com.conveyal.r5.transit.path.RouteSequence;
Expand All @@ -16,7 +15,6 @@
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.awt.*;
import java.lang.invoke.MethodHandles;
import java.util.ArrayList;
import java.util.Arrays;
Expand Down Expand Up @@ -59,6 +57,13 @@ public class PathResult {
*/
public final Multimap<RouteSequence, Iteration>[] iterationsForPathTemplates;

/**
* The total number of iterations that reached each destination can be derived from iterationsForPathTemplates
* as long as every path is being retained. When filtering down to a subset of paths, such as only those passing
* through a selected link, we need this additional array to retain the information.
*/
private final int[] nUnfilteredIterationsReachingDestination;

private final TransitLayer transitLayer;

public static final String[] DATA_COLUMNS = new String[]{
Expand Down Expand Up @@ -86,7 +91,9 @@ public PathResult(AnalysisWorkerTask task, TransitLayer transitLayer) {
throw new UnsupportedOperationException("Number of detailed path destinations exceeds limit of " + MAX_PATH_DESTINATIONS);
}
}
// FIXME should we be allocating these large arrays when not recording paths?
iterationsForPathTemplates = new Multimap[nDestinations];
nUnfilteredIterationsReachingDestination = new int[nDestinations];
this.transitLayer = transitLayer;
}

Expand All @@ -95,10 +102,13 @@ public PathResult(AnalysisWorkerTask task, TransitLayer transitLayer) {
* pattern-based keys
*/
public void setTarget(int targetIndex, Multimap<PatternSequence, Iteration> patterns) {
// The size of a multimap is the number of mappings (number of values), not number of unique keys.
// This size method appears to be O(1), see: com.google.common.collect.AbstractMapBasedMultimap.size
nUnfilteredIterationsReachingDestination[targetIndex] = patterns.size();

// When selected link analysis is enabled, filter down the PatternSequence-Iteration Multimap to retain only
// those keys passing through the selected links.
// TODO Maybe selectedLink should be on TransitLayer, and somehow indicate the number of removed iterations.
// Maybe selectedLink instance should be on TransitLayer not TransportNetwork.
if (transitLayer.parentNetwork.selectedLink != null) {
patterns = transitLayer.parentNetwork.selectedLink.filterPatterns(patterns);
}
Expand All @@ -125,6 +135,7 @@ public ArrayList<String[]>[] summarizeIterations(Stat stat) {
Multimap<RouteSequence, Iteration> iterationMap = iterationsForPathTemplates[d];
if (iterationMap != null) {
// SelectedLink case: collapse all RouteSequences and Iterations for this OD pair into one to simplify.
// iterationMap is empty (not null) for destinations that were reached without using the selected link.
// This could also be done by merging all Iterations under a single RouteSequence with all route IDs.
if (transitLayer.parentNetwork.selectedLink != null) {
int nIterations = 0;
Expand All @@ -137,20 +148,27 @@ public ArrayList<String[]>[] summarizeIterations(Stat stat) {
summedTotalTime += iterations.stream().mapToInt(i -> i.totalTime).sum();
}
// Many destinations will have no iterations at all passing through the SelectedLink area.
// Skip those to keep the CSV output short.
if (nIterations > 0) {
String[] row = new String[DATA_COLUMNS.length];
Arrays.fill(row, "ALL");
String allRouteIdsPipeSeparated = Arrays.stream(allRouteIds.toArray())
.mapToObj(transitLayer.routes::get)
.map(routeInfo -> routeInfo.route_id)
.collect(Collectors.joining("|"));
row[0] = allRouteIdsPipeSeparated;
row[row.length - 1] = Integer.toString(nIterations);
row[row.length - 2] = String.format("%.1f", summedTotalTime / nIterations / 60d); // Average total time
summary[d].add(row);
// Skip those to keep the CSV output short (and to avoid division by zero below).
if (nIterations == 0) {
continue;
}
continue;
String[] row = new String[DATA_COLUMNS.length];
Arrays.fill(row, "ALL");
transitLayer.routeString(1, true);
String allRouteIdsPipeSeparated = Arrays.stream(allRouteIds.toArray())
// If includeName is set to false we record only the ID without the name.
// Name works better than ID for routes added by modifications, which have random IDs.
.mapToObj(ri -> transitLayer.routeString(ri, true))
.collect(Collectors.joining("|"));
String iterationProportion = "%.3f".formatted(
nIterations / (double)(nUnfilteredIterationsReachingDestination[d]));
row[0] = allRouteIdsPipeSeparated;
row[row.length - 1] = iterationProportion;
// Report average of total time over all retained iterations, different than mean/min approach below.
row[row.length - 2] = String.format("%.1f", summedTotalTime / nIterations / 60d);
summary[d].add(row);
// Fall through to the standard case below, so the summary row is followed by its component parts.
// We could optionally continue to the next loop iteration here, to return only the summary row.
}
// Standard (non SelectedLink) case.
for (RouteSequence routeSequence: iterationMap.keySet()) {
Expand Down Expand Up @@ -185,7 +203,11 @@ public ArrayList<String[]>[] summarizeIterations(Stat stat) {
score = thisScore;
}
}
String[] row = ArrayUtils.addAll(path, transfer, waits, totalTime, String.valueOf(nIterations));
// Check above guarantees that nIterations is nonzero, so total iterations must be nonzero,
// avoiding divide by zero.
String iterationProportion = "%.3f".formatted(
nIterations / (double)(nUnfilteredIterationsReachingDestination[d]));
String[] row = ArrayUtils.addAll(path, transfer, waits, totalTime, iterationProportion);
checkState(row.length == DATA_COLUMNS.length);
summary[d].add(row);
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -126,7 +126,7 @@ public boolean apply(TransportNetwork network) {
patterns.size(), nPatternsWithoutShapes
));
}
// After finding all links (TripPattern hops) in the SelectedLink area, release the GTFSFeeds which don't really
// After finding all links (TripPattern hops) in the SelectedLink area, release the GTFSFeeds. They don't really
// belong in a Modification. This avoids memory leaks, and protects us from inadvertently relying on or
// modifying those feed objects later.
feedForUnscopedId = null;
Expand All @@ -140,12 +140,13 @@ public boolean apply(TransportNetwork network) {
.map(s -> tripPattern.stops[s])
.mapToObj(network.transitLayer.stopNames::get)
.collect(Collectors.joining(", "));
String message = String.format("Route %s direction %s after stop %s", routeInfo.getName(), tripPattern.directionId, stopNames);
// Report route name here rather than ID, as the name is better defined on routes created by modifications.
String message = String.format("Route %s direction %s after stop %s",
routeInfo.getName(), tripPattern.directionId, stopNames);
addInfo(message);
LOG.info(message);
return true;
});

// Store the resulting precomputed information in a SelectedLink instance on the TransportNetwork.
// This could also be on the TransitLayer, but we may eventually want to include street edges in SelectedLink.
network.selectedLink = new SelectedLink(network.transitLayer, hopsInTripPattern);
Expand Down

0 comments on commit a4bf241

Please sign in to comment.