Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Dont abort bidirectional a* search if only one direction is exhausted #2936

Merged
merged 8 commits into from
Mar 23, 2021

Conversation

mandeepsandhu
Copy link
Contributor

Instead, we force search in the remaining direction and fail if both
directions exhaust or one of the terminating conditions become true.

forward_exhausted = true;
LOG_WARN("Forward search exhausted: n = " + std::to_string(edgelabels_forward_.size()) +
"," + std::to_string(edgelabels_reverse_.size()));
} else {
Copy link
Member

@kevinkreiser kevinkreiser Mar 12, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this is a nitpick but i feel like at this point we should flip the logic around to be like:

if (!best_connections_.empty()) {
  return FormPath(graphreader, options, origin, destination, forward_time_info, invariant);
}
// mark the forward direction exhausted so we dont expand from here anymore
forward_exhausted = true;
LOG_WARN("Forward search exhausted: n = " + std::to_string(edgelabels_forward_.size()) + "," + std::to_string(edgelabels_reverse_.size()));

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

will do

test::customize_edges(map.config, [&not_thru_edgeids](const GraphId& edgeid, DirectedEdge& edge) {
if (std::find(not_thru_edgeids.begin(), not_thru_edgeids.end(), edgeid) !=
not_thru_edgeids.end()) {
edge.set_not_thru(true);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

its great to see you make use of that! i thought that one day opening that up for testing would be useful!

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I do not think this is a valid test case. I believe that not_thru will only be set in one direction. A directed edge that enters an area with one entrance/exit is marked as not_thru but the opposing edge that exits that area will not be marked as not_thru. Perhaps the not_thru logic is too aggressive that it somehow is marking some edges as not_thru in both directions, but the design is that only one direction of an edge should ever be marked as not_thru.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i agree that its not true to what will actually get created in the data but its handy to be able to force the router into this situation even though its contrived. it would be better to show that it naturally happens for derived things like not_through, but at the same time the you have to figure out and spend tons of time designing the layout of the test to get it to happen.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

oh you have no idea how much time I spent on trying to recreate a scenario in gurka which would mark an edhe as not_thru! Thx to @genadz for pointing me to an existing test which was using this method.

BTW, customize_edges does not work if we're also setting live traffic conditions on the edge. When using live traffic, we pass-in a cleanly (I believe to avoid caching) created "reader" which does not have the customizations created in customize_edges. I think it has to do with live traffic & customize_edges altering different in-memory representations of the tile (I have to dig in further for the exact root-cause though).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I do not think this is a valid test case. I believe that not_thru will only be set in one direction. A directed edge that enters an area with one entrance/exit is marked as not_thru but the opposing edge that exits that area will not be marked as not_thru. Perhaps the not_thru logic is too aggressive that it somehow is marking some edges as not_thru in both directions, but the design is that only one direction of an edge should ever be marked as not_thru.

You're right, this is a contrived scenario with not thru's. However, such a scenario can happen when routing out of closures (edges marked closed due to live traffic conditions). In this case, the reverse search will not be able to expand into a closure.

I originally hit this condition while testing something with closures. ButI wanted to keep the test simple without adding live traffic into the mix, so resorted to making a test with not_thru's.

Comment on lines +716 to +750
// Expand from the search direction with lower sort cost
// Note: If one direction is exhausted, we force search in the remaining
// direction
Copy link
Member

@kevinkreiser kevinkreiser Mar 12, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: why do your comments seem to have a shorter line break than your code? 🤔

Copy link
Contributor Author

@mandeepsandhu mandeepsandhu Mar 12, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Because once upon a time (long before twitter made 120-char limits fashionable), 80-char limit was the common std for line-wrapping. I still haven't fully shrugged off that habit 😄 . I'll fix this in the next PR update.

@kevinkreiser
Copy link
Member

kevinkreiser commented Mar 12, 2021

this code change looks good to solve the issue we face where we can find a route but dont. however we need to understand what this does to routes that can't find a path and shouldn't.

i'm not sure if i was making sense to you over in #2926 (comment) but one of the things the code in master does now is if one side of the search is trapped in an island then the code bails fast (as soon as that side is exhausted). this is a good thing. for routes that should fail, we want them to fail as fast as possible. perhaps you can take a corpus of failed routes and run it through this branch to see how often they fail slower than before (because they wait until we do too many iterations in the other search direction instead of bailing immediately). ie long request time just to get back a no route is wasted CPU

@@ -622,6 +622,8 @@ BidirectionalAStar::GetBestPath(valhalla::Location& origin,
BDEdgeLabel fwd_pred, rev_pred;
bool expand_forward = true;
bool expand_reverse = true;
bool forward_exhausted = false;
bool reverse_exhausted = false;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why do we need these additional variables ? we can check if forward_pred_idx equals to kInvalidLabel or not

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We could (i initially used that in one of my initial iterations), but found the bools a bit more readable and make the intent more explicit.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Perhaps a best of both worlds could be bool is_forward_exhausted() { return "forward_pred_idx == kInvalidLabel" }?

@dnesbitt61
Copy link
Member

dnesbitt61 commented Mar 12, 2021

ditto on @kevinkreiser comment about failing fast. I have seen cases where nearly the entire NA road network is expanded when unable to actually reach the other search direction. That can hang one thread in a server for a considerable time. I still argue that if one direction's search is exhausted without any connections being found then the route is not possible. Also, won't a second pass be generated that would relax conditions that might lead to a failure (hierarchy, not_thru)?

@dnesbitt61
Copy link
Member

Note that another case is when a data set is an extract (e.g. tiles within a bounding box for an embedded Valhalla implementation). In this case there can be regions that are disconnected from each other and this relies on quick failures (exhausting one direction).

@mandeepsandhu
Copy link
Contributor Author

ditto on @kevinkreiser comment about failing fast. I have seen cases where nearly the entire NA road network is expanded when unable to actually reach the other search direction. That can hang one thread in a server for a considerable time. I still argue that if one direction's search is exhausted without any connections being found then the route is not possible. Also, won't a second pass be generated that would relax conditions that might lead to a failure (hierarchy, not_thru)?

I think that concern (about no-route situations taking longer to complete) is valid. I'm not yet sure what other criteria can be used to abort (or extend) search from the other direction if one direction is exhausted (maybe dynamically change one of the termination thresholds like cost or number of iterations if one side exhausts?).

As for the second pass, not all restrictions are relaxed/avoided. Eg not thru's and traffic closures are not ignored in the second pass (so the route would still fail).

As @kevinkreiser asked above, let me get some numbers on time it takes for the no-route routes before/after this change. I'm open to dropping this change if the no-route take significantly longer and don't justify finding routes for very specific cases (routing out of small islands with high cost edges).

@mandeepsandhu
Copy link
Contributor Author

this code change looks good to solve the issue we face where we can find a route but dont. however we need to understand what this does to routes can't find a path and shouldn't.

i'm not sure if i was making sense to you over in #2926 (comment) but one of the things the code in master does now is if one side of the search is trapped in an island then the code bails fast (as soon as that side is exhausted). this is a good thing. for routes that should fail, we want them to fail as fast as possible. perhaps you can take a corpus of failed routes and run it through this branch to see how often they fail slower than before (because they wait until we do too many iterations in the other search direction instead of bailing immediately). ie long request time just to get back a no route is wasted CPU

I agree. Let me get some numbers for you and we can discuss it from there.

@purew
Copy link
Contributor

purew commented Mar 12, 2021

I'm not yet sure what other criteria can be used to abort (or extend) search from the other direction if one direction is exhausted (maybe dynamically change one of the termination thresholds like cost or number of iterations if one side exhausts?).

Can you look at if the expansion from the non-exhausted side is making progress towards the destination or simply expanding further and further away from the destination?

@mandeepsandhu
Copy link
Contributor Author

mandeepsandhu commented Mar 12, 2021

I'm not yet sure what other criteria can be used to abort (or extend) search from the other direction if one direction is exhausted (maybe dynamically change one of the termination thresholds like cost or number of iterations if one side exhausts?).

Can you look at if the expansion from the non-exhausted side is making progress towards the destination or simply expanding further and further away from the destination?

I'm not yet sure what other criteria can be used to abort (or extend) search from the other direction if one direction is exhausted (maybe dynamically change one of the termination thresholds like cost or number of iterations if one side exhausts?).

Can you look at if the expansion from the non-exhausted side is making progress towards the destination or simply expanding further and further away from the destination?

I can imagine some cases where even that might not be sufficient. eg doing a cross-country route across US, where start point is on the west coast and the destination lies in an unconnected island on the east coast. So the forward search, will be moving towards the destination for the most part (not many edges to explore in the other direction) and will continue for quite some time

maybe if we can detect the side that finished was part of a small connected component then we can abort prematurely.

@mandeepsandhu
Copy link
Contributor Author

Here are the performance numbers using de_benchmark_routes.txt:

percentile 50th 90th 99th
master 0.0516 0.0941 0.1625
branch 0.0607 (+17.52%) 0.1126 (+19.7%) 2.508 (1440%)

Dumping the iteration count for a failed route shows how many more we edges are searched:

{"locations": [{"lat":26.596227982772817,"lon":-80.0191682907218},{"lat":26.49913918981666,"lon":-78.414660088092}], "costing":"auto"}

master:

2021/03/13 00:17:38.681467 [ERROR] Bi-directional route failure - reverse search exhausted: n = 3090,26086
2021/03/13 00:17:38.712002 [ERROR] Bi-directional route failure - reverse search exhausted: n = 12116,77343

branch (# fwd edges, # rev edges):

021/03/13 00:18:13.193237 [ERROR] Bi-directional route failure - search exhausted: n = 1197679,3090
2021/03/13 00:18:13.630715 [ERROR] Bi-directional route failure - search exhausted: n = 1248798,12116

In both cases the number of edges in reverse direction is the same (3090), but the edges searched in the fwd direction is ~50 times more (in the first pass).

I also tested a few working routes from the test file, and both master and this branch showed the same number of edges explored. I'm not sure whats contributing to the difference in 50th percentile as I'd expect the impact to be only in the no-route scenario.

Either way, extending the search clearly has a big negative impact. Given this, I'm hesitant to move this forward with this PR, unless there's a better way to reduce the search space for no-route situations.

@kevinkreiser @dnesbitt61 thoughts?

@dnesbitt61
Copy link
Member

I cannot think of a way to effectively limit the expansion short of breaking out after some number of iterations.

My view is that it is best to fail quickly in this case, at least on the first pass of the algorithm.

@kevinkreiser
Copy link
Member

@mandeepsandhu i think something must be wrong here if im understanding you correctly. i dont know how many routes fail in the test set but it sounded like you didnt separate routes that fail from routes that didnt. i would first want to confirm that routes that dont fail see no performance impact. if they do (which it seemed like your numbers suggest) then we need to figure out why as this change should not affect that!

then once that is worked out i would suggest we sequester the failing routes into their own file and see what performance diff we see on them. over all, since we have such a low route failure rate, an increase in time to fail might be tolerable. we just need to separate the two cases so we can understand what the code change is doing to each.

again i expected no change to non-failure routes unless they were retry (2 pass) routes or something.

@genadz
Copy link
Contributor

genadz commented Mar 15, 2021

I cannot think of a way to effectively limit the expansion short of breaking out after some number of iterations.

My view is that it is best to fail quickly in this case, at least on the first pass of the algorithm.

@mandeepsandhu @kevinkreiser @dnesbitt61 I have an idea: right now there are 2 types of "not_thru" regions: marked as "not_thru" and marked as "closed" due to live traffic. So, in case one of the searches exhausts we should check if another one is still in "not_thru" region. If it's true, we should continue another search until we find a connection.

@kevinkreiser
Copy link
Member

@genadz i was saying that before in either the issue or a resolved comment. we would need to track whether expansion stopped because it hit not through regions on all paths or not. that seemed expensive to me. maybe you are saying if the other side is expanding in not through or closed, so tracking from the other search path?

@mandeepsandhu
Copy link
Contributor Author

@mandeepsandhu i think something must be wrong here if im understanding you correctly. i dont know how many routes fail in the test set but it sounded like you didnt separate routes that fail from routes that didnt. i would first want to confirm that routes that dont fail see no performance impact. if they do (which it seemed like your numbers suggest) then we need to figure out why as this change should not affect that!

Okay, let me wean out the failing routes and see how many of those are there. Spot checking individual non-failing routes, as expected, I didn't see a difference in performance and the number of labels explored in fwd/rev direction were exactly the same between master & this branch (but these were just 3-4 out of the 2500 routes).

then once that is worked out i would suggest we sequester the failing routes into their own file and see what performance diff we see on them. over all, since we have such a low route failure rate, an increase in time to fail might be tolerable. we just need to separate the two cases so we can understand what the code change is doing to each.

again i expected no change to non-failure routes unless they were retry (2 pass) routes or something.

okay, I'll get back to you after I've removed the failing routes and measured them separately.

@mandeepsandhu
Copy link
Contributor Author

I cannot think of a way to effectively limit the expansion short of breaking out after some number of iterations.
My view is that it is best to fail quickly in this case, at least on the first pass of the algorithm.

@mandeepsandhu @kevinkreiser @dnesbitt61 I have an idea: right now there are 2 types of "not_thru" regions: marked as "not_thru" and marked as "closed" due to live traffic. So, in case one of the searches exhausts we should check if another one is still in "not_thru" region. If it's true, we should continue another search until we find a connection.

Yeah we could use the state of not_thru/closure pruning to decide if we want to continue expanding in that direction if the other direction has exhausted. However, it could happen that the label on top of the priority queue when the other direction's search exhausts, does not have pruning disabled (maybe the edges with pruning disabled are slightly higher cost and therefore have not been expanded from yet). Maybe you're suggesting that if any edge in the currently explored set of labels has pruning disabled, we should continue with the search?

@kevinkreiser
Copy link
Member

However, it could happen that the label on top of the priority queue when the other direction's search exhausts, does not have pruning disabled

hahaha i was saying exactly this to gena an hour or so ago 😄

@genadz
Copy link
Contributor

genadz commented Mar 15, 2021

However, it could happen that the label on top of the priority queue when the other direction's search exhausts, does not have pruning disabled

hahaha i was saying exactly this to gena an hour or so ago

It may happen only when search that was exhausted explored not all adjacency edge to "not_thru" region. if we want to handle this too we should check if we have unexplored edges from "not_thru" region. it requires additional logic to maintain number of unexplored "not_thru" edges, so probably not worth of it

@mandeepsandhu
Copy link
Contributor Author

mandeepsandhu commented Mar 16, 2021

So I separated out the routes in de_benchmark_routes.txt:
that were taking long (there were 36 of them). These were failing in the 1st pass, and succeeding with relaxed parameters in the second pass.

@kevinkreiser as you expected, the non-2pass routes were un-touched in performance.

Fast routes:

percentile 50th 90th 99th
master 0.0522 0.0931 0.1632
branch 0.0520 (-0.08%) 0.0927 (-0.44%) 0.1631 (-0.41%)

Slow 2-pass routes:

percentile 50th 90th 99th
master 0.0977 0.1406 0.2068
branch 2.928 3.137 3.198

Example of 2-pass route:

'{"costing":"auto","directions_options":{"units":"kilometers"},"locations":[{"city":"Frankfurt am Main","lat":50.111591,"lon":8.681067},{"city":"Koeln","lat":50.937962,"lon":6.960059}]}'

We can look into conditionally expanding the search (when 1 side exhausts) based on if any of the expansions from the other side is within a closure/not_thru region or not. This will require some additional bookkeeping. I'm still debating if the added complexity is worth it or not. I'm also okay dropping this altogether and just live with certain route failures (these happen in very specific circumstances), unless you know of other legitimate cases where we'd like to extend the search. Let me know what you think @dnesbitt61 @danpat @genadz @kevinkreiser.

@mandeepsandhu
Copy link
Contributor Author

mandeepsandhu commented Mar 16, 2021

Thinking about this a bit more, we have the following options:

  1. Keep current behaviour i.e fail route if one side of the search exhausts
  2. Keep track of pruning states (for closures and not_thrus) and extend the search iff there's at least 1 expansion which has pruning disabled. This requires keeping the closure/not_thru state of each branch of expansion and will be non-trivial to implement. But gives the most thorough solution.
  3. Instead of keeping track of pruning states of all branches of expansion, extend search if the other end started behind a closed or not_thru region. This should be trivial to implement, but is not robust to all cases. This will handle majority of the no-route cases, expect impossible routes that start behind not_thrus & closures.
  4. Change the iteration or cost threshold if one side of the search has been exhausted, so we don't expand for too long before aborting. This sounds a bit iffy, and I'm not sure what the new values for the thresholds should be such that it allows legitimate cases to succeed while failing the others in reasonable time.
  5. Keep this implementation and live with the added penalty of extending search in one direction.

Any other solution that we can think of?
(update: added another option - 4)

@dnesbitt61
Copy link
Member

My preference would be to keep the current behavior on the first pass (fail quickly). This way if there is a closure and a path is found that avoids the closure it will be returned. I am assuming that not_thru behavior is not the main problem being solved here (let me know if you have found cases where not_thru logic alone is failing routes).

On a subsequent pass, adjust closures to allow them (perhaps with a high penalty) and identify if the successful route used a closure. This way, the client can identify routes that succeed but with a closure taken - e.g. indicate the only way to complete the route is using the closure.

@mandeepsandhu
Copy link
Contributor Author

mandeepsandhu commented Mar 16, 2021

My preference would be to keep the current behavior on the first pass (fail quickly). This way if there is a closure and a path is found that avoids the closure it will be returned. I am assuming that not_thru behavior is not the main problem being solved here (let me know if you have found cases where not_thru logic alone is failing routes).

On a subsequent pass, adjust closures to allow them (perhaps with a high penalty) and identify if the successful route used a closure. This way, the client can identify routes that succeed but with a closure taken - e.g. indicate the only way to complete the route is using the closure.

quick recap of how routing through closures work:

  • Valhalla has a user selectable option to start/end routes within closures. This is enabled by setting search_filter:exclude_closures to false (it defaults to true) for a waypoint location.
  • What this does is it allows for origin and destination edges to be snapped to closed roads. It also allows use of other consecutive closed edges when doing the a* expansion, but does NOT ignore them if they are in the middle of the route (i.e closures which are not connected to either origin or destination edge via other consecutive closures). Closed edges are currently not penalized negatively.

The problem surfaced when I started penalizing the closures (so that the route can get out of it asap), since it forces search from only direction until the costs become equal. Although this is not restricted to closures, and the following 2 routes hit the exhaustion due to search hitting a not_thru region (theres no live traffic being used here):

In your solution, we could possibly set ignore_closures: true (via costing options) in the second-pass, although this flag will cause us to ignore ALL closures and not just those at the origin/dest. To prevent the router from taking intermediate closures, we could penalize the closure heavily which will prevent intermediate closures from being used, but allow those at origin/dest to be used (as those are the only edges that can reach the locations). Is that what you had in mind?

@dnesbitt61
Copy link
Member

dnesbitt61 commented Mar 17, 2021

Surprised at the not_thru cases. I tested these 2 cases with Valhalla data from earlier this year and they both succeed. This data does not have hierarchies and shortcuts and has driveways removed (it was intended for bicycles).

I'll stay out of the closures discussion as I haven't thought enough about it to make any recommendations.

My main point is that if you make these changes to extend searches, please make them conditional so that existing applications do not suddenly have long route times for some unexpected cases. I especially worry about the region extract case (mobile applications) where locations near the edge of the extract may be truly unreachable.

@kevinkreiser
Copy link
Member

I personally like options number 3 above where we only extend it in a very special circumstance. Also now that we do allow configuration passed into the algorithm, I think we can do what @dnesbitt61 is suggesting and make it something you can opt out of via a config parameter. what do you think about that @mandeepsandhu ?

@mandeepsandhu
Copy link
Contributor Author

mandeepsandhu commented Mar 17, 2021

I personally like options number 3 above where we only extend it in a very special circumstance. Also now that we do allow configuration passed into the algorithm, I think we can do what @dnesbitt61 is suggesting and make it something you can opt out of via a config parameter. what do you think about that @mandeepsandhu ?

@kevinkreiser +1 to that. To recap and make sure we're all on the same page:

  • Create a valhalla (thor) config to decide if we want to extend search if one side exhausts
  • Extend the search only if the other end is behind a closure or not_thru region

correct?

@kevinkreiser
Copy link
Member

correct?

@mandeepsandhu that would be my suggestion yep!

@mandeepsandhu
Copy link
Contributor Author

Performance numbers with latest master vs branch:

With option to extend search disabled (thor.extended_search: false), should be similar to master:

de_benchmark_routes.txt (full set of routes)

percentile 50th 90th 99th
master 0.0554 0.0992 0.1697
branch 0.0542 0.0979 0.1682

Fast routes with extended search enabled (thor.extended_search: true):

percentile 50th 90th 99th
branch 0.0548 0.0962 0.1686

Slow 2-pass routes with extended search enabled (thor.extended_search: true):

percentile 50th 90th 99th
branch 0.106 0.260 1.648

CC: @danpat @dnesbitt61 @genadz @kevinkreiser

Copy link
Contributor

@purew purew left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM but I'll leave final approve to @kevinkreiser or @dnesbitt61 as they've been more active in the PR.

@kevinkreiser
Copy link
Member

Fast routes with extended search enabled

this means one that find a path yes?

Slow 2-pass routes with extended search enabled

this means ones that fail on the first pass and also fail on the second pass but now fail slower or do some of them pass now? or maybe we dont have any test routes that exhibit the failure scenario that is seeking to handle?

@mandeepsandhu
Copy link
Contributor Author

Fast routes with extended search enabled

this means one that find a path yes?

Yes, in 1 pass.

Slow 2-pass routes with extended search enabled

this means ones that fail on the first pass and also fail on the second pass but now fail slower or do some of them pass now? or maybe we dont have any test routes that exhibit the failure scenario that is seeking to handle?

There are no route failures. The slow 2-pass ones are the ones which fail in the first pass (search exhaustion in one direction) and then the second pass with relaxed restrictions passes. With extended search, its the first pass failure that takes longer (if the other end is behind not_thru).

So there are no route failures finally, just that some specific 2-pass routes are slow now due to extended search.

@kevinkreiser
Copy link
Member

ok then one final question, can you run the full set with it enabled. i dont have a sense of how frequently we have to do a 2 pass approach so i cant tell how much overal numbers will be pulled toward the performance drop. i mean i love that its configurable so its a bit moot but its good to know! also im not sure how representative these DE routes are

@mandeepsandhu
Copy link
Contributor Author

ok then one final question, can you run the full set with it enabled. i dont have a sense of how frequently we have to do a 2 pass approach so i cant tell how much overal numbers will be pulled toward the performance drop. i mean i love that its configurable so its a bit moot but its good to know! also im not sure how representative these DE routes are

There are 36 routes in de_benchmark_routes.txt that all exhaust their search, and succeed in the second pass. I was able to extract out these 36 routes and replay them separately (thats what the "slow 2-pass" results are).

Here's the per comparison with master vs branch with extended search enabled:

de_benchmark_routes.txt (full set of routes)

percentile 50th 90th 99th
master 0.0546 0.0977 0.1698
branch 0.0555 (+1.8%) 0.0989 (+1.2) 0.1793 (+5.6%)

Here's another comparison of a set of "1k" test routes that I have locally, to hopefully give a more representative view of routes:

percentile 50th 90th 99th
master 0.0267 0.1006 0.4302
branch 0.0281 (+5.24%) 0.1085 (+7.85) 0.4657 (+8.2%)

...and one more corpus of 20k routes:

percentile 50th 90th 99th
master 0.0399 0.0799 0.2056
branch 0.0398 (-0.25%) 0.0804 (+0.62%) 0.2053 (-0.14%)

(branch being faster is likely due to measurement noise)

pa_interchange_routes.txt:

percentile 50th 90th 99th
master 0.0142 0.0212 0.0240
branch 0.0132 (-7%) 0.0176 (-16.98%) 0.0227 (-5.4%)

bicycle_routes.txt:

percentile 50th 90th 99th
master 0.0134 0.0518 0.0946
branch 0.0139 (+3.73%) 0.0524 (+1.15%) 0.0787 (-16.8%)

kevinkreiser
kevinkreiser previously approved these changes Mar 22, 2021
Copy link
Member

@kevinkreiser kevinkreiser left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

looks good to me, especially because its opt in. i do wonder if for closure we should have done the same thing we did with not through on the second pass and just allow them. are we thinking about that in the future?

@mandeepsandhu
Copy link
Contributor Author

looks good to me, especially because its opt in. i do wonder if for closure we should have done the same thing we did with not through on the second pass and just allow them. are we thinking about that in the future?

@kevinkreiser do you mean we should set costng -> ignore_closures : true in the second pass? (this will ignore all closures)

Test case illustrates the case where 1 directions of bidirectional
expansion being exhausted causes route failures.
Instead, we force search in the remaining direction and fail if both
directions exhaust or one of the terminating conditions become true.
@mandeepsandhu mandeepsandhu merged commit 614fb03 into master Mar 23, 2021
abhi2039 pushed a commit to abhi2039/valhalla that referenced this pull request Mar 30, 2021
…halla into pedestrian_crossing

* 'pedestrian_crossing' of https://github.com/abhi2039/valhalla:
  build: Bail with error if non-existant build-type specified (valhalla#2965)
  nit: Enables compiler warnings in part of mjolnir module (valhalla#2922)
  fix missing comma scripts/valhalla_build_config (valhalla#2963)
  Remove duplicate names being used across different gurka tests (valhalla#2962)
  Dont abort bidirectional a* search if only one direction is exhausted (valhalla#2936)
  penalize uturns at pencil points and short internal edges (valhalla#2944)
  Remove unused IsNextEdgeInternal function (valhalla#2958)
  restore compatibility with gcc 6.3.0, libprotobuf 3.0.0, boost v1.62.0 (valhalla#2953)
  Add ability to build Valhalla modules as STATIC libraries (valhalla#2957)
  Allow disabling Werror (valhalla#2937)
  Enhanced logic for IsTurnChannelManeuverCombinable (valhalla#2952)
  Allow config object to be passed-in to path algorithms (valhalla#2949)
  Floating-point precision fix in meili::Project() fixes map-match issue (valhalla#2946)
  Use kServiceRoad edges while reclassifying ferry connections (valhalla#2933)
  ci: Update cache key to trigger clearing it (valhalla#2947)
@nilsnolde nilsnolde deleted the expansion_exhaustion branch February 24, 2024 15:08
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants