From d6cd1e13f800269b6328626782d70d3efa3d8830 Mon Sep 17 00:00:00 2001 From: Michael Tsang Date: Mon, 9 Dec 2024 16:52:08 +0000 Subject: [PATCH 1/5] parallel raptor cache generation --- .../transit/RaptorTransferIndex.java | 35 +++++++++++-------- 1 file changed, 21 insertions(+), 14 deletions(-) diff --git a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java index 1d9b804067c..f8d0cf1f696 100644 --- a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java +++ b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java @@ -5,6 +5,7 @@ import java.util.ArrayList; import java.util.List; import java.util.function.Function; +import java.util.stream.IntStream; import org.opentripplanner.raptor.api.model.RaptorTransfer; import org.opentripplanner.street.search.request.StreetSearchRequest; @@ -35,27 +36,33 @@ public static RaptorTransferIndex create( reversedTransfers.add(new ArrayList<>()); } - for (int fromStop = 0; fromStop < transfersByStopIndex.size(); fromStop++) { - // The transfers are filtered so that there is only one possible directional transfer - // for a stop pair. - var transfers = transfersByStopIndex - .get(fromStop) - .stream() - .flatMap(s -> s.asRaptorTransfer(request).stream()) - .collect( - toMap(RaptorTransfer::stop, Function.identity(), (a, b) -> a.c1() < b.c1() ? a : b) - ) - .values(); + IntStream + .range(0, transfersByStopIndex.size()) + .parallel() + .forEach(fromStop -> { + // The transfers are filtered so that there is only one possible directional transfer + // for a stop pair. + var transfers = transfersByStopIndex + .get(fromStop) + .stream() + .flatMap(s -> s.asRaptorTransfer(request).stream()) + .collect( + toMap(RaptorTransfer::stop, Function.identity(), (a, b) -> a.c1() < b.c1() ? a : b) + ) + .values(); - forwardTransfers.get(fromStop).addAll(transfers); + // forwardTransfers is not modified here, and no two threads will access the same element + // in it, so this is still thread safe. + forwardTransfers.get(fromStop).addAll(transfers); + }); - for (RaptorTransfer forwardTransfer : transfers) { + for (int fromStop = 0; fromStop < transfersByStopIndex.size(); fromStop++) { + for (var forwardTransfer : forwardTransfers.get(fromStop)) { reversedTransfers .get(forwardTransfer.stop()) .add(DefaultRaptorTransfer.reverseOf(fromStop, forwardTransfer)); } } - return new RaptorTransferIndex(forwardTransfers, reversedTransfers); } From 1bf3f655ab25bb7194d8d0980665c58d34b4f359 Mon Sep 17 00:00:00 2001 From: Michael Tsang Date: Thu, 12 Dec 2024 13:43:09 +0000 Subject: [PATCH 2/5] add feature check around the parallel operation --- .../transit/RaptorTransferIndex.java | 38 ++++++++++--------- 1 file changed, 20 insertions(+), 18 deletions(-) diff --git a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java index f8d0cf1f696..aa335939086 100644 --- a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java +++ b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java @@ -6,6 +6,7 @@ import java.util.List; import java.util.function.Function; import java.util.stream.IntStream; +import org.opentripplanner.framework.application.OTPFeature; import org.opentripplanner.raptor.api.model.RaptorTransfer; import org.opentripplanner.street.search.request.StreetSearchRequest; @@ -36,25 +37,26 @@ public static RaptorTransferIndex create( reversedTransfers.add(new ArrayList<>()); } - IntStream - .range(0, transfersByStopIndex.size()) - .parallel() - .forEach(fromStop -> { - // The transfers are filtered so that there is only one possible directional transfer - // for a stop pair. - var transfers = transfersByStopIndex - .get(fromStop) - .stream() - .flatMap(s -> s.asRaptorTransfer(request).stream()) - .collect( - toMap(RaptorTransfer::stop, Function.identity(), (a, b) -> a.c1() < b.c1() ? a : b) - ) - .values(); + var stopIndices = IntStream.range(0, transfersByStopIndex.size()); + if (OTPFeature.ParallelRouting.isOn()) { + stopIndices = stopIndices.parallel(); + } + stopIndices.forEach(fromStop -> { + // The transfers are filtered so that there is only one possible directional transfer + // for a stop pair. + var transfers = transfersByStopIndex + .get(fromStop) + .stream() + .flatMap(s -> s.asRaptorTransfer(request).stream()) + .collect( + toMap(RaptorTransfer::stop, Function.identity(), (a, b) -> a.c1() < b.c1() ? a : b) + ) + .values(); - // forwardTransfers is not modified here, and no two threads will access the same element - // in it, so this is still thread safe. - forwardTransfers.get(fromStop).addAll(transfers); - }); + // forwardTransfers is not modified here, and no two threads will access the same element + // in it, so this is still thread safe. + forwardTransfers.get(fromStop).addAll(transfers); + }); for (int fromStop = 0; fromStop < transfersByStopIndex.size(); fromStop++) { for (var forwardTransfer : forwardTransfers.get(fromStop)) { From cded8677ecdae7553713395976c5502b20e41661 Mon Sep 17 00:00:00 2001 From: Michael Tsang Date: Thu, 12 Dec 2024 15:16:49 +0000 Subject: [PATCH 3/5] Always parallelize cache building during server startup --- .../raptoradapter/transit/RaptorTransferIndex.java | 13 +++++++++++-- .../transit/request/RaptorRequestTransferCache.java | 5 +++-- 2 files changed, 14 insertions(+), 4 deletions(-) diff --git a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java index aa335939086..61ece117388 100644 --- a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java +++ b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java @@ -25,9 +25,16 @@ public RaptorTransferIndex( this.reversedTransfers = reversedTransfers.stream().map(List::copyOf).toArray(List[]::new); } + /** + * Create an index to be put into the transfer cache + * + * @param isRuntimeRequest true if the request originates from the client during the runtime, + * false if the request comes from transferCacheRequests in router-config.json + */ public static RaptorTransferIndex create( List> transfersByStopIndex, - StreetSearchRequest request + StreetSearchRequest request, + boolean isRuntimeRequest ) { var forwardTransfers = new ArrayList>(transfersByStopIndex.size()); var reversedTransfers = new ArrayList>(transfersByStopIndex.size()); @@ -38,7 +45,9 @@ public static RaptorTransferIndex create( } var stopIndices = IntStream.range(0, transfersByStopIndex.size()); - if (OTPFeature.ParallelRouting.isOn()) { + // we want to always parallelize the cache building during the startup + // and only parallelize during runtime requests if the feature flag is on + if (!isRuntimeRequest || OTPFeature.ParallelRouting.isOn()) { stopIndices = stopIndices.parallel(); } stopIndices.forEach(fromStop -> { diff --git a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/request/RaptorRequestTransferCache.java b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/request/RaptorRequestTransferCache.java index d778f491142..c7e09e12729 100644 --- a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/request/RaptorRequestTransferCache.java +++ b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/request/RaptorRequestTransferCache.java @@ -38,7 +38,8 @@ public void put(List> transfersByStopIndex, RouteRequest request) final CacheKey cacheKey = new CacheKey(transfersByStopIndex, request); final RaptorTransferIndex raptorTransferIndex = RaptorTransferIndex.create( transfersByStopIndex, - cacheKey.request + cacheKey.request, + false ); LOG.info("Initializing cache with request: {}", cacheKey.options); @@ -58,7 +59,7 @@ private CacheLoader cacheLoader() { @Override public RaptorTransferIndex load(CacheKey cacheKey) { LOG.info("Adding runtime request to cache: {}", cacheKey.options); - return RaptorTransferIndex.create(cacheKey.transfersByStopIndex, cacheKey.request); + return RaptorTransferIndex.create(cacheKey.transfersByStopIndex, cacheKey.request, true); } }; } From 082dafbb55077f7f49613f8585f339b32be80dee Mon Sep 17 00:00:00 2001 From: Michael Tsang Date: Thu, 19 Dec 2024 15:31:42 +0000 Subject: [PATCH 4/5] use an enum to specify the request source --- .../raptoradapter/transit/RaptorTransferIndex.java | 7 ++----- .../raptoradapter/transit/RequestSource.java | 12 ++++++++++++ .../transit/request/RaptorRequestTransferCache.java | 9 +++++++-- 3 files changed, 21 insertions(+), 7 deletions(-) create mode 100644 application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RequestSource.java diff --git a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java index eaa472214b2..d034a9d8e35 100644 --- a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java +++ b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java @@ -28,14 +28,11 @@ public RaptorTransferIndex( /** * Create an index to be put into the transfer cache - * - * @param isRuntimeRequest true if the request originates from the client during the runtime, - * false if the request comes from transferCacheRequests in router-config.json */ public static RaptorTransferIndex create( List> transfersByStopIndex, StreetSearchRequest request, - boolean isRuntimeRequest + RequestSource requestSource ) { var forwardTransfers = new ArrayList>(transfersByStopIndex.size()); var reversedTransfers = new ArrayList>(transfersByStopIndex.size()); @@ -49,7 +46,7 @@ public static RaptorTransferIndex create( var stopIndices = IntStream.range(0, transfersByStopIndex.size()); // we want to always parallelize the cache building during the startup // and only parallelize during runtime requests if the feature flag is on - if (!isRuntimeRequest || OTPFeature.ParallelRouting.isOn()) { + if (requestSource == RequestSource.CONFIG || OTPFeature.ParallelRouting.isOn()) { stopIndices = stopIndices.parallel(); } stopIndices.forEach(fromStop -> { diff --git a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RequestSource.java b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RequestSource.java new file mode 100644 index 00000000000..9e49d2da78e --- /dev/null +++ b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RequestSource.java @@ -0,0 +1,12 @@ +package org.opentripplanner.routing.algorithm.raptoradapter.transit; + +public enum RequestSource { + /** + * The request comes from transferCacheRequests in router-config.json + */ + CONFIG, + /** + * The request comes from a client routing request + */ + RUNTIME, +} diff --git a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/request/RaptorRequestTransferCache.java b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/request/RaptorRequestTransferCache.java index c7e09e12729..23724e0dd34 100644 --- a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/request/RaptorRequestTransferCache.java +++ b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/request/RaptorRequestTransferCache.java @@ -7,6 +7,7 @@ import java.util.Objects; import java.util.concurrent.ExecutionException; import org.opentripplanner.routing.algorithm.raptoradapter.transit.RaptorTransferIndex; +import org.opentripplanner.routing.algorithm.raptoradapter.transit.RequestSource; import org.opentripplanner.routing.algorithm.raptoradapter.transit.Transfer; import org.opentripplanner.routing.api.request.RouteRequest; import org.opentripplanner.routing.api.request.StreetMode; @@ -39,7 +40,7 @@ public void put(List> transfersByStopIndex, RouteRequest request) final RaptorTransferIndex raptorTransferIndex = RaptorTransferIndex.create( transfersByStopIndex, cacheKey.request, - false + RequestSource.CONFIG ); LOG.info("Initializing cache with request: {}", cacheKey.options); @@ -59,7 +60,11 @@ private CacheLoader cacheLoader() { @Override public RaptorTransferIndex load(CacheKey cacheKey) { LOG.info("Adding runtime request to cache: {}", cacheKey.options); - return RaptorTransferIndex.create(cacheKey.transfersByStopIndex, cacheKey.request, true); + return RaptorTransferIndex.create( + cacheKey.transfersByStopIndex, + cacheKey.request, + RequestSource.RUNTIME + ); } }; } From 2ae88164d7d2ebb90778b0a87df8fce39e6af9b5 Mon Sep 17 00:00:00 2001 From: Michael Tsang Date: Fri, 3 Jan 2025 10:30:42 +0000 Subject: [PATCH 5/5] apply review feedback --- .../transit/RaptorTransferIndex.java | 28 +++++++++++++++++-- .../raptoradapter/transit/RequestSource.java | 12 -------- .../request/RaptorRequestTransferCache.java | 11 +++----- 3 files changed, 29 insertions(+), 22 deletions(-) delete mode 100644 application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RequestSource.java diff --git a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java index d034a9d8e35..fa44d793664 100644 --- a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java +++ b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RaptorTransferIndex.java @@ -13,6 +13,11 @@ public class RaptorTransferIndex { + private enum RequestSource { + SETUP, + REQUEST_SCOPE, + } + private final List[] forwardTransfers; private final List[] reversedTransfers; @@ -27,9 +32,26 @@ public RaptorTransferIndex( } /** - * Create an index to be put into the transfer cache + * Create an index for a route request configured in router-config.json + */ + public static RaptorTransferIndex createInitialSetup( + List> transfersByStopIndex, + StreetSearchRequest request + ) { + return create(transfersByStopIndex, request, RequestSource.SETUP); + } + + /** + * Create an index for a route request originated from the client */ - public static RaptorTransferIndex create( + public static RaptorTransferIndex createRequestScope( + List> transfersByStopIndex, + StreetSearchRequest request + ) { + return create(transfersByStopIndex, request, RequestSource.REQUEST_SCOPE); + } + + private static RaptorTransferIndex create( List> transfersByStopIndex, StreetSearchRequest request, RequestSource requestSource @@ -46,7 +68,7 @@ public static RaptorTransferIndex create( var stopIndices = IntStream.range(0, transfersByStopIndex.size()); // we want to always parallelize the cache building during the startup // and only parallelize during runtime requests if the feature flag is on - if (requestSource == RequestSource.CONFIG || OTPFeature.ParallelRouting.isOn()) { + if (requestSource == RequestSource.SETUP || OTPFeature.ParallelRouting.isOn()) { stopIndices = stopIndices.parallel(); } stopIndices.forEach(fromStop -> { diff --git a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RequestSource.java b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RequestSource.java deleted file mode 100644 index 9e49d2da78e..00000000000 --- a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/RequestSource.java +++ /dev/null @@ -1,12 +0,0 @@ -package org.opentripplanner.routing.algorithm.raptoradapter.transit; - -public enum RequestSource { - /** - * The request comes from transferCacheRequests in router-config.json - */ - CONFIG, - /** - * The request comes from a client routing request - */ - RUNTIME, -} diff --git a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/request/RaptorRequestTransferCache.java b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/request/RaptorRequestTransferCache.java index 23724e0dd34..80814fdeee2 100644 --- a/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/request/RaptorRequestTransferCache.java +++ b/application/src/main/java/org/opentripplanner/routing/algorithm/raptoradapter/transit/request/RaptorRequestTransferCache.java @@ -7,7 +7,6 @@ import java.util.Objects; import java.util.concurrent.ExecutionException; import org.opentripplanner.routing.algorithm.raptoradapter.transit.RaptorTransferIndex; -import org.opentripplanner.routing.algorithm.raptoradapter.transit.RequestSource; import org.opentripplanner.routing.algorithm.raptoradapter.transit.Transfer; import org.opentripplanner.routing.api.request.RouteRequest; import org.opentripplanner.routing.api.request.StreetMode; @@ -37,10 +36,9 @@ public LoadingCache getTransferCache() { public void put(List> transfersByStopIndex, RouteRequest request) { final CacheKey cacheKey = new CacheKey(transfersByStopIndex, request); - final RaptorTransferIndex raptorTransferIndex = RaptorTransferIndex.create( + final RaptorTransferIndex raptorTransferIndex = RaptorTransferIndex.createInitialSetup( transfersByStopIndex, - cacheKey.request, - RequestSource.CONFIG + cacheKey.request ); LOG.info("Initializing cache with request: {}", cacheKey.options); @@ -60,10 +58,9 @@ private CacheLoader cacheLoader() { @Override public RaptorTransferIndex load(CacheKey cacheKey) { LOG.info("Adding runtime request to cache: {}", cacheKey.options); - return RaptorTransferIndex.create( + return RaptorTransferIndex.createRequestScope( cacheKey.transfersByStopIndex, - cacheKey.request, - RequestSource.RUNTIME + cacheKey.request ); } };