Shortest path planning
Planning shortest paths in Cypher® can lead to different query plans depending on the predicates that need
to be evaluated. Internally, Neo4j will use a fast bidirectional breadth-first search algorithm if the
predicates can be evaluated whilst searching for the path. Therefore, this fast algorithm will always
be certain to return the right answer when there are universal predicates on the path; for example, when
searching for the shortest path where all nodes have the Person label, or where there are no nodes with
a name property.
If the predicates need to inspect the whole path before deciding on whether it is valid or not, this fast
algorithm cannot be relied on to find the shortest path, and Neo4j may have to resort to using a slower
exhaustive depth-first search algorithm to find the path. This means that query plans for shortest path
queries with non-universal predicates will include a fallback to running the exhaustive search to find
the path should the fast algorithm not succeed. For example, depending on the data, an answer to a shortest
path query with existential predicates — such as the requirement that at least one node contains the property
name='Kevin Bacon' — may not be able to be found by the fast algorithm. In this case, Neo4j will fall back to using
the exhaustive search to enumerate all paths and potentially return an answer.
The running times of these two algorithms may differ by orders of magnitude, so it is important to ensure that the fast approach is used for time-critical queries.
When the exhaustive search is planned, it is still only executed when the fast algorithm fails to find any matching paths. The fast algorithm is always executed first, since it is possible that it can find a valid path even though that could not be guaranteed at planning time.
Please note that falling back to the exhaustive search may prove to be a very time consuming strategy in some
cases; such as when there is no shortest path between two nodes.
Therefore, in these cases, it is recommended to set cypher.forbid_exhaustive_shortestpath to true,
as explained in Operations Manual → Configuration settings
Shortest path with fast algorithm
MATCH (KevinB:Person {name: 'Kevin Bacon'} ),
      (Al:Person {name: 'Al Pacino'}),
      p = shortestPath((KevinB)-[:ACTED_IN*]-(Al))
WHERE all(r IN relationships(p) WHERE exists(r.role))
RETURN pThis query can be evaluated with the fast algorithm — there are no predicates that need to see the whole path before being evaluated.
Compiler CYPHER 4.2
Planner COST
Runtime PIPELINED
Runtime version 4.2
+---------------------+----------------------------------------------------------------------------------------------+----------------+------+---------+----------------+------------------------+-----------+---------------+
| Operator            | Details                                                                                      | Estimated Rows | Rows | DB Hits | Memory (Bytes) | Page Cache Hits/Misses | Time (ms) | Other         |
+---------------------+----------------------------------------------------------------------------------------------+----------------+------+---------+----------------+------------------------+-----------+---------------+
| +ProduceResults     | p                                                                                            |              0 |    1 |       0 |                |                    0/0 |     1.165 | In Pipeline 1 |
| |                   +----------------------------------------------------------------------------------------------+----------------+------+---------+----------------+------------------------+-----------+---------------+
| +ShortestPath       | p = (KevinB)-[anon_117:ACTED_IN*]-(Al) WHERE all(r IN relationships(p) WHERE exists(r.role)) |              0 |    1 |      13 |           1704 |                        |           | In Pipeline 1 |
| |                   +----------------------------------------------------------------------------------------------+----------------+------+---------+----------------+------------------------+-----------+---------------+
| +MultiNodeIndexSeek | KevinB:Person(name) WHERE name = $autostring_0, Al:Person(name) WHERE name = $autostring_1   |              0 |    1 |       4 |             72 |                    1/1 |    14.057 | In Pipeline 0 |
+---------------------+----------------------------------------------------------------------------------------------+----------------+------+---------+----------------+------------------------+-----------+---------------+
Total database accesses: 17, total allocated memory: 1704Shortest path with additional predicate checks on the paths
Consider using the exhaustive search as a fallback
Predicates used in the WHERE clause that apply to the shortest path pattern are evaluated before deciding
what the shortest matching path is.
MATCH (KevinB:Person {name: 'Kevin Bacon'}),
      (Al:Person {name: 'Al Pacino'}),
      p = shortestPath((KevinB)-[*]-(Al))
WHERE length(p) > 1
RETURN pThis query, in contrast with the one above, needs to check that the whole path follows the predicate before we know if it is valid or not, and so the query plan will also include the fallback to the slower exhaustive search algorithm.
Compiler CYPHER 4.2
Planner COST
Runtime SLOTTED
Runtime version 4.2
+--------------------------+------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| Operator                 | Details                                                    | Estimated Rows | Rows | DB Hits | Memory (Bytes) | Page Cache Hits/Misses |
+--------------------------+------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| +ProduceResults          | p                                                          |              0 |    1 |       0 |                |                    0/0 |
| |                        +------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| +AntiConditionalApply    |                                                            |              0 |    1 |       0 |                |                    0/0 |
| |\                       +------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| | +Top                   | anon_94 ASC LIMIT 1                                        |              0 |    0 |       0 |                |                    0/0 |
| | |                      +------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| | +Projection            | length(p) AS anon_94                                       |              0 |    0 |       0 |                |                    0/0 |
| | |                      +------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| | +Filter                | length(p) > $autoint_2                                     |              0 |    0 |       0 |                |                    0/0 |
| | |                      +------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| | +Projection            | (KevinB)-[anon_116*]-(Al) AS p                             |              0 |    0 |       0 |                |                    0/0 |
| | |                      +------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| | +VarLengthExpand(Into) | (KevinB)-[anon_116*]-(Al)                                  |              0 |    0 |       0 |                |                    0/0 |
| | |                      +------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| | +Argument              | KevinB, Al, p, anon_116                                    |              0 |    0 |       0 |                |                    0/0 |
| |                        +------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| +Apply                   |                                                            |              0 |    1 |       0 |                |                    0/0 |
| |\                       +------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| | +Optional              | KevinB, Al                                                 |              1 |    1 |       0 |                |                    0/0 |
| | |                      +------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| | +ShortestPath          | p = (KevinB)-[anon_116*]-(Al) WHERE length(p) > $autoint_2 |              0 |    1 |       1 |           1776 |                   21/0 |
| | |                      +------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| | +Argument              | KevinB, Al                                                 |              0 |    1 |       0 |                |                    0/0 |
| |                        +------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| +CartesianProduct        |                                                            |              0 |    1 |       0 |                |                    0/0 |
| |\                       +------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| | +NodeIndexSeek         | Al:Person(name) WHERE name = $autostring_1                 |              0 |    1 |       2 |                |                    1/0 |
| |                        +------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
| +NodeIndexSeek           | KevinB:Person(name) WHERE name = $autostring_0             |              0 |    1 |       2 |                |                    1/0 |
+--------------------------+------------------------------------------------------------+----------------+------+---------+----------------+------------------------+
Total database accesses: 5, total allocated memory: 1776The way the bigger exhaustive query plan works is by using Apply/Optional to ensure that when the
fast algorithm does not find any results, a null result is generated instead of simply stopping the result
stream.
On top of this, the planner will issue an AntiConditionalApply, which will run the exhaustive search
if the path variable is pointing to null instead of a path.
An ErrorPlan operator will appear in the execution plan in cases where (i)
cypher.forbid_exhaustive_shortestpath is set to true, and (ii) the fast algorithm is not able to find the shortest path.
Prevent the exhaustive search from being used as a fallback
MATCH (KevinB:Person {name: 'Kevin Bacon'}),
      (Al:Person {name: 'Al Pacino'}),
      p = shortestPath((KevinB)-[*]-(Al))
WITH p
WHERE length(p) > 1
RETURN pThis query, just like the one above, needs to check that the whole path follows the predicate
before we know if it is valid or not. However, the inclusion of the WITH clause means that the query
plan will not include the fallback to the slower exhaustive search algorithm. Instead, any
paths found by the fast algorithm will subsequently be filtered, which may result in no answers
 being returned.
Compiler CYPHER 4.2
Planner COST
Runtime PIPELINED
Runtime version 4.2
+---------------------+--------------------------------------------------------------------------------------------+----------------+------+---------+----------------+------------------------+-----------+---------------+
| Operator            | Details                                                                                    | Estimated Rows | Rows | DB Hits | Memory (Bytes) | Page Cache Hits/Misses | Time (ms) | Other         |
+---------------------+--------------------------------------------------------------------------------------------+----------------+------+---------+----------------+------------------------+-----------+---------------+
| +ProduceResults     | p                                                                                          |              1 |    1 |       0 |                |                    0/0 |     0.112 | In Pipeline 1 |
| |                   +--------------------------------------------------------------------------------------------+----------------+------+---------+----------------+------------------------+-----------+---------------+
| +Filter             | length(p) > $autoint_2                                                                     |              0 |    1 |       0 |                |                    0/0 |     0.093 | In Pipeline 1 |
| |                   +--------------------------------------------------------------------------------------------+----------------+------+---------+----------------+------------------------+-----------+---------------+
| +ShortestPath       | p = (KevinB)-[anon_116*]-(Al)                                                              |              0 |    1 |       1 |           1776 |                        |           | In Pipeline 1 |
| |                   +--------------------------------------------------------------------------------------------+----------------+------+---------+----------------+------------------------+-----------+---------------+
| +MultiNodeIndexSeek | KevinB:Person(name) WHERE name = $autostring_0, Al:Person(name) WHERE name = $autostring_1 |              0 |    1 |       4 |             72 |                    2/0 |     0.221 | In Pipeline 0 |
+---------------------+--------------------------------------------------------------------------------------------+----------------+------+---------+----------------+------------------------+-----------+---------------+
Total database accesses: 5, total allocated memory: 1776