k Shortest Paths in AQL

General query idea

This type of query finds the first k paths in order of length (or weight) between two given documents, startVertex and targetVertex in your graph.

Every such path will be returned as a JSON object with three components:

  • an array containing the vertices on the path
  • an array containing the edges on the path
  • the weight of the path, that is the sum of all edge weights

If no weightAttribute is given, the weight of the path is just its length.

Example

Let us take a look at a simple example to explain how it works. This is the graph that we are going to find some shortest path on:

Train Connection Map

Each ellipse stands for a train station with the name of the city written inside of it. They are the vertices of the graph. Arrows represent train connections between cities and are the edges of the graph. The numbers near the arrows describe how long it takes to get from one station to another. They are used as edge weights.

Let us assume that we want to go from Aberdeen to London by train.

We expect to see the following vertices on the shortest path, in this order:

  1. Aberdeen
  2. Leuchars
  3. Edinburgh
  4. York
  5. London

By the way, the weight of the path is: 1.5 + 1.5 + 3.5 + 1.8 = 8.3.

Let us look at alternative paths next, for example because we know that the direct connection between York and London does not operate currently. An alternative path, which is slightly longer, goes like this:

  1. Aberdeen
  2. Leuchars
  3. Edinburgh
  4. York
  5. Carlisle
  6. Birmingham
  7. London

Its weight is: 1.5 + 1.5 + 3.5 + 2.0 + 1.5 = 10.0.

Another route goes via Glasgow. There are seven stations on the path as well, however, it is quicker if we compare the edge weights:

  1. Aberdeen
  2. Leuchars
  3. Edinburgh
  4. Glasgow
  5. Carlisle
  6. Birmingham
  7. London

The path weight is lower: 1.5 + 1.5 + 1.0 + 1.0 + 2.0 + 1.5 = 8.5.

Syntax

The syntax for k Shortest Paths queries is similar to the one for Shortest Path and there are also two options to either use a named graph or a set of edge collections. It only emits a path variable however, whereas SHORTEST_PATH emits a vertex and an edge variable.

It is highly recommended that you use a LIMIT statement, as k Shortest Paths is a potentially expensive operation. On large connected graphs it can return a large number of paths, or perform an expensive (but unsuccessful) search for more short paths.

Working with named graphs

FOR path
  IN OUTBOUND|INBOUND|ANY K_SHORTEST_PATHS
  startVertex TO targetVertex
  GRAPH graphName
  [OPTIONS options]
  [LIMIT offset, count]
  • FOR: emits the variable path which contains one path as an object containing vertices, edges, and the weight of the path.
  • IN OUTBOUND|INBOUND|ANY: defines in which direction edges are followed (outgoing, incoming, or both)
  • K_SHORTEST_PATHS: the keyword to compute k Shortest Paths
  • startVertex TO targetVertex (both string|object): the two vertices between which the paths will be computed. This can be specified in the form of a ID string or in the form of a document with the attribute _id. All other values will lead to a warning and an empty result. If one of the specified documents does not exist, the result is empty as well and there is no warning.
  • GRAPH graphName (string): the name identifying the named graph. Its vertex and edge collections will be looked up.
  • OPTIONS options (object, optional): used to modify the execution of the traversal. Only the following attributes have an effect, all others are ignored:
    • weightAttribute (string): a top-level edge attribute that should be used to read the edge weight. If the attribute does not exist or is not numeric, the defaultWeight will be used instead. The attribute value must not be negative.
    • defaultWeight (number): this value will be used as fallback if there is no weightAttribute in the edge document, or if it’s not a number. The value must not be negative. The default is 1.
  • LIMIT (see LIMIT operation, optional): the maximal number of paths to return. It is highly recommended to use a LIMIT for K_SHORTEST_PATHS.

k Shortest Paths traversals do not support negative weights. If a document attribute (as specified by weightAttribute) with a negative value is encountered during traversal, or if defaultWeight is set to a negative number, then the query is aborted with an error.

Working with collection sets

FOR path
  IN OUTBOUND|INBOUND|ANY K_SHORTEST_PATHS
  startVertex TO targetVertex
  edgeCollection1, ..., edgeCollectionN
  [OPTIONS options]
  [LIMIT offset, count]

Instead of GRAPH graphName you can specify a list of edge collections. The involved vertex collections are determined by the edges of the given edge collections.

Traversing in mixed directions

For k shortest paths with a list of edge collections you can optionally specify the direction for some of the edge collections. Say for example you have three edge collections edges1, edges2 and edges3, where in edges2 the direction has no relevance, but in edges1 and edges3 the direction should be taken into account. In this case you can use OUTBOUND as general search direction and ANY specifically for edges2 as follows:

FOR vertex IN OUTBOUND K_SHORTEST_PATHS
  startVertex TO targetVertex
  edges1, ANY edges2, edges3

All collections in the list that do not specify their own direction will use the direction defined after IN (here: OUTBOUND). This allows to use a different direction for each collection in your path search.

Examples

We load an example graph to get a named graph that reflects some possible train connections in Europe and North America.

Train Connection Map

arangosh> var examples = require("@arangodb/graph-examples/example-graph.js");
arangosh> var graph = examples.loadGraph("kShortestPathsGraph");
arangosh> db.places.toArray();
arangosh> db.connections.toArray();
Show execution results
Hide execution results
[ 
  { 
    "_key" : "Inverness", 
    "_id" : "places/Inverness", 
    "_rev" : "_cuv8msa---", 
    "label" : "Inverness" 
  }, 
  { 
    "_key" : "Aberdeen", 
    "_id" : "places/Aberdeen", 
    "_rev" : "_cuv8msa--_", 
    "label" : "Aberdeen" 
  }, 
  { 
    "_key" : "Leuchars", 
    "_id" : "places/Leuchars", 
    "_rev" : "_cuv8mse---", 
    "label" : "Leuchars" 
  }, 
  { 
    "_key" : "StAndrews", 
    "_id" : "places/StAndrews", 
    "_rev" : "_cuv8mse--_", 
    "label" : "StAndrews" 
  }, 
  { 
    "_key" : "Edinburgh", 
    "_id" : "places/Edinburgh", 
    "_rev" : "_cuv8msi---", 
    "label" : "Edinburgh" 
  }, 
  { 
    "_key" : "Glasgow", 
    "_id" : "places/Glasgow", 
    "_rev" : "_cuv8msi--_", 
    "label" : "Glasgow" 
  }, 
  { 
    "_key" : "York", 
    "_id" : "places/York", 
    "_rev" : "_cuv8msi--A", 
    "label" : "York" 
  }, 
  { 
    "_key" : "Carlisle", 
    "_id" : "places/Carlisle", 
    "_rev" : "_cuv8msi--B", 
    "label" : "Carlisle" 
  }, 
  { 
    "_key" : "Birmingham", 
    "_id" : "places/Birmingham", 
    "_rev" : "_cuv8msm---", 
    "label" : "Birmingham" 
  }, 
  { 
    "_key" : "London", 
    "_id" : "places/London", 
    "_rev" : "_cuv8msm--_", 
    "label" : "London" 
  }, 
  { 
    "_key" : "Brussels", 
    "_id" : "places/Brussels", 
    "_rev" : "_cuv8msm--A", 
    "label" : "Brussels" 
  }, 
  { 
    "_key" : "Cologne", 
    "_id" : "places/Cologne", 
    "_rev" : "_cuv8msq---", 
    "label" : "Cologne" 
  }, 
  { 
    "_key" : "Toronto", 
    "_id" : "places/Toronto", 
    "_rev" : "_cuv8msq--_", 
    "label" : "Toronto" 
  }, 
  { 
    "_key" : "Winnipeg", 
    "_id" : "places/Winnipeg", 
    "_rev" : "_cuv8msu---", 
    "label" : "Winnipeg" 
  }, 
  { 
    "_key" : "Saskatoon", 
    "_id" : "places/Saskatoon", 
    "_rev" : "_cuv8msu--_", 
    "label" : "Saskatoon" 
  }, 
  { 
    "_key" : "Edmonton", 
    "_id" : "places/Edmonton", 
    "_rev" : "_cuv8msu--A", 
    "label" : "Edmonton" 
  }, 
  { 
    "_key" : "Jasper", 
    "_id" : "places/Jasper", 
    "_rev" : "_cuv8msy---", 
    "label" : "Jasper" 
  }, 
  { 
    "_key" : "Vancouver", 
    "_id" : "places/Vancouver", 
    "_rev" : "_cuv8msy--_", 
    "label" : "Vancouver" 
  } 
]
[ 
  { 
    "_key" : "61880", 
    "_id" : "connections/61880", 
    "_from" : "places/Inverness", 
    "_to" : "places/Aberdeen", 
    "_rev" : "_cuv8ms2---", 
    "travelTime" : 3 
  }, 
  { 
    "_key" : "61882", 
    "_id" : "connections/61882", 
    "_from" : "places/Aberdeen", 
    "_to" : "places/Inverness", 
    "_rev" : "_cuv8ms2--_", 
    "travelTime" : 2.5 
  }, 
  { 
    "_key" : "61884", 
    "_id" : "connections/61884", 
    "_from" : "places/Aberdeen", 
    "_to" : "places/Leuchars", 
    "_rev" : "_cuv8ms6---", 
    "travelTime" : 1.5 
  }, 
  { 
    "_key" : "61886", 
    "_id" : "connections/61886", 
    "_from" : "places/Leuchars", 
    "_to" : "places/Aberdeen", 
    "_rev" : "_cuv8ms6--_", 
    "travelTime" : 1 
  }, 
  { 
    "_key" : "61888", 
    "_id" : "connections/61888", 
    "_from" : "places/Leuchars", 
    "_to" : "places/Edinburgh", 
    "_rev" : "_cuv8mt----", 
    "travelTime" : 1.5 
  }, 
  { 
    "_key" : "61890", 
    "_id" : "connections/61890", 
    "_from" : "places/Edinburgh", 
    "_to" : "places/Leuchars", 
    "_rev" : "_cuv8mt---_", 
    "travelTime" : 3 
  }, 
  { 
    "_key" : "61892", 
    "_id" : "connections/61892", 
    "_from" : "places/Edinburgh", 
    "_to" : "places/Glasgow", 
    "_rev" : "_cuv8mtC---", 
    "travelTime" : 1 
  }, 
  { 
    "_key" : "61894", 
    "_id" : "connections/61894", 
    "_from" : "places/Glasgow", 
    "_to" : "places/Edinburgh", 
    "_rev" : "_cuv8mtC--_", 
    "travelTime" : 1 
  }, 
  { 
    "_key" : "61896", 
    "_id" : "connections/61896", 
    "_from" : "places/Edinburgh", 
    "_to" : "places/York", 
    "_rev" : "_cuv8mtC--A", 
    "travelTime" : 3.5 
  }, 
  { 
    "_key" : "61898", 
    "_id" : "connections/61898", 
    "_from" : "places/York", 
    "_to" : "places/Edinburgh", 
    "_rev" : "_cuv8mtC--B", 
    "travelTime" : 4 
  }, 
  { 
    "_key" : "61900", 
    "_id" : "connections/61900", 
    "_from" : "places/Glasgow", 
    "_to" : "places/Carlisle", 
    "_rev" : "_cuv8mtG---", 
    "travelTime" : 1 
  }, 
  { 
    "_key" : "61902", 
    "_id" : "connections/61902", 
    "_from" : "places/Carlisle", 
    "_to" : "places/Glasgow", 
    "_rev" : "_cuv8mtG--_", 
    "travelTime" : 1 
  }, 
  { 
    "_key" : "61904", 
    "_id" : "connections/61904", 
    "_from" : "places/Carlisle", 
    "_to" : "places/York", 
    "_rev" : "_cuv8mtG--A", 
    "travelTime" : 2.5 
  }, 
  { 
    "_key" : "61906", 
    "_id" : "connections/61906", 
    "_from" : "places/York", 
    "_to" : "places/Carlisle", 
    "_rev" : "_cuv8mtK---", 
    "travelTime" : 3.5 
  }, 
  { 
    "_key" : "61908", 
    "_id" : "connections/61908", 
    "_from" : "places/Carlisle", 
    "_to" : "places/Birmingham", 
    "_rev" : "_cuv8mtK--_", 
    "travelTime" : 2 
  }, 
  { 
    "_key" : "61910", 
    "_id" : "connections/61910", 
    "_from" : "places/Birmingham", 
    "_to" : "places/Carlisle", 
    "_rev" : "_cuv8mtK--A", 
    "travelTime" : 1 
  }, 
  { 
    "_key" : "61912", 
    "_id" : "connections/61912", 
    "_from" : "places/Birmingham", 
    "_to" : "places/London", 
    "_rev" : "_cuv8mtO---", 
    "travelTime" : 1.5 
  }, 
  { 
    "_key" : "61914", 
    "_id" : "connections/61914", 
    "_from" : "places/London", 
    "_to" : "places/Birmingham", 
    "_rev" : "_cuv8mtO--_", 
    "travelTime" : 2.5 
  }, 
  { 
    "_key" : "61916", 
    "_id" : "connections/61916", 
    "_from" : "places/Leuchars", 
    "_to" : "places/StAndrews", 
    "_rev" : "_cuv8mtS---", 
    "travelTime" : 0.2 
  }, 
  { 
    "_key" : "61918", 
    "_id" : "connections/61918", 
    "_from" : "places/StAndrews", 
    "_to" : "places/Leuchars", 
    "_rev" : "_cuv8mtS--_", 
    "travelTime" : 0.2 
  }, 
  { 
    "_key" : "61920", 
    "_id" : "connections/61920", 
    "_from" : "places/York", 
    "_to" : "places/London", 
    "_rev" : "_cuv8mtW---", 
    "travelTime" : 1.8 
  }, 
  { 
    "_key" : "61922", 
    "_id" : "connections/61922", 
    "_from" : "places/London", 
    "_to" : "places/York", 
    "_rev" : "_cuv8mtW--_", 
    "travelTime" : 2 
  }, 
  { 
    "_key" : "61924", 
    "_id" : "connections/61924", 
    "_from" : "places/London", 
    "_to" : "places/Brussels", 
    "_rev" : "_cuv8mta---", 
    "travelTime" : 2.5 
  }, 
  { 
    "_key" : "61926", 
    "_id" : "connections/61926", 
    "_from" : "places/Brussels", 
    "_to" : "places/London", 
    "_rev" : "_cuv8mta--_", 
    "travelTime" : 3.5 
  }, 
  { 
    "_key" : "61928", 
    "_id" : "connections/61928", 
    "_from" : "places/Brussels", 
    "_to" : "places/Cologne", 
    "_rev" : "_cuv8mte---", 
    "travelTime" : 2 
  }, 
  { 
    "_key" : "61930", 
    "_id" : "connections/61930", 
    "_from" : "places/Cologne", 
    "_to" : "places/Brussels", 
    "_rev" : "_cuv8mte--_", 
    "travelTime" : 1.5 
  }, 
  { 
    "_key" : "61932", 
    "_id" : "connections/61932", 
    "_from" : "places/Toronto", 
    "_to" : "places/Winnipeg", 
    "_rev" : "_cuv8mti---", 
    "travelTime" : 36 
  }, 
  { 
    "_key" : "61934", 
    "_id" : "connections/61934", 
    "_from" : "places/Winnipeg", 
    "_to" : "places/Toronto", 
    "_rev" : "_cuv8mti--_", 
    "travelTime" : 35 
  }, 
  { 
    "_key" : "61936", 
    "_id" : "connections/61936", 
    "_from" : "places/Winnipeg", 
    "_to" : "places/Saskatoon", 
    "_rev" : "_cuv8mtm---", 
    "travelTime" : 12 
  }, 
  { 
    "_key" : "61938", 
    "_id" : "connections/61938", 
    "_from" : "places/Saskatoon", 
    "_to" : "places/Winnipeg", 
    "_rev" : "_cuv8mtm--_", 
    "travelTime" : 5 
  }, 
  { 
    "_key" : "61940", 
    "_id" : "connections/61940", 
    "_from" : "places/Saskatoon", 
    "_to" : "places/Edmonton", 
    "_rev" : "_cuv8mtm--A", 
    "travelTime" : 12 
  }, 
  { 
    "_key" : "61942", 
    "_id" : "connections/61942", 
    "_from" : "places/Edmonton", 
    "_to" : "places/Saskatoon", 
    "_rev" : "_cuv8mtq---", 
    "travelTime" : 17 
  }, 
  { 
    "_key" : "61944", 
    "_id" : "connections/61944", 
    "_from" : "places/Edmonton", 
    "_to" : "places/Jasper", 
    "_rev" : "_cuv8mtq--_", 
    "travelTime" : 6 
  }, 
  { 
    "_key" : "61946", 
    "_id" : "connections/61946", 
    "_from" : "places/Jasper", 
    "_to" : "places/Edmonton", 
    "_rev" : "_cuv8mtu---", 
    "travelTime" : 5 
  }, 
  { 
    "_key" : "61948", 
    "_id" : "connections/61948", 
    "_from" : "places/Jasper", 
    "_to" : "places/Vancouver", 
    "_rev" : "_cuv8mtu--_", 
    "travelTime" : 12 
  }, 
  { 
    "_key" : "61950", 
    "_id" : "connections/61950", 
    "_from" : "places/Vancouver", 
    "_to" : "places/Jasper", 
    "_rev" : "_cuv8mty---", 
    "travelTime" : 13 
  } 
]

Suppose we want to query a route from Aberdeen to London, and compare the outputs of SHORTEST_PATH and K_SHORTEST_PATHS with LIMIT 1. Note that while SHORTEST_PATH and K_SHORTEST_PATH with LIMIT 1 should return a path of the same length (or weight), they do not need to return the same path.

Using SHORTEST_PATH:

FOR v, e IN OUTBOUND SHORTEST_PATH 'places/Aberdeen' TO 'places/London'
  GRAPH 'kShortestPathsGraph'
      RETURN { place: v.label, travelTime: e.travelTime }
Show query results
Hide query results
[
  {
    "place": "Aberdeen",
    "travelTime": null
  },
  {
    "place": "Leuchars",
    "travelTime": 1.5
  },
  {
    "place": "Edinburgh",
    "travelTime": 1.5
  },
  {
    "place": "York",
    "travelTime": 3.5
  },
  {
    "place": "London",
    "travelTime": 1.8
  }
]

Using K_SHORTEST_PATHS:

FOR p IN OUTBOUND K_SHORTEST_PATHS 'places/Aberdeen' TO 'places/London'
  GRAPH 'kShortestPathsGraph'
      LIMIT 1
      RETURN { places: p.vertices[*].label, travelTimes: p.edges[*].travelTime }
Show query results
Hide query results
[
  {
    "places": [
      "Aberdeen",
      "Leuchars",
      "Edinburgh",
      "York",
      "London"
    ],
    "travelTimes": [
      1.5,
      1.5,
      3.5,
      1.8
    ]
  }
]

With K_SHORTEST_PATHS we can ask for more than one option for a route:

FOR p IN OUTBOUND K_SHORTEST_PATHS 'places/Aberdeen' TO 'places/London'
  GRAPH 'kShortestPathsGraph'
      LIMIT 3
      RETURN {
          places: p.vertices[*].label,
          travelTimes: p.edges[*].travelTime,
          travelTimeTotal: SUM(p.edges[*].travelTime)
      }
Show query results
Hide query results
[
  {
    "places": [
      "Aberdeen",
      "Leuchars",
      "Edinburgh",
      "York",
      "London"
    ],
    "travelTimes": [
      1.5,
      1.5,
      3.5,
      1.8
    ],
    "travelTimeTotal": 8.3
  },
  {
    "places": [
      "Aberdeen",
      "Leuchars",
      "Edinburgh",
      "York",
      "Carlisle",
      "Birmingham",
      "London"
    ],
    "travelTimes": [
      1.5,
      1.5,
      3.5,
      3.5,
      2,
      1.5
    ],
    "travelTimeTotal": 13.5
  },
  {
    "places": [
      "Aberdeen",
      "Leuchars",
      "Edinburgh",
      "Glasgow",
      "Carlisle",
      "York",
      "London"
    ],
    "travelTimes": [
      1.5,
      1.5,
      1,
      1,
      2.5,
      1.8
    ],
    "travelTimeTotal": 9.3
  }
]

If we ask for routes that don’t exist we get an empty result (from Aberdeen to Toronto):

FOR p IN OUTBOUND K_SHORTEST_PATHS 'places/Aberdeen' TO 'places/Toronto'
  GRAPH 'kShortestPathsGraph'
      LIMIT 3
      RETURN {
          places: p.vertices[*].label,
          travelTimes: p.edges[*].travelTime,
          travelTimeTotal: SUM(p.edges[*].travelTime)
      }
Show query results
Hide query results
[]

We can use the attribute travelTime that connections have as edge weights to take into account which connections are quicker. A high default weight is set, to be used if an edge has no travelTime attribute (not the case with the example graph). This returns the top three routes with the fewest changes and favoring the least travel time for the connection Saint Andrews to Cologne:

FOR p IN OUTBOUND K_SHORTEST_PATHS 'places/StAndrews' TO 'places/Cologne'
  GRAPH 'kShortestPathsGraph'
  OPTIONS {
      weightAttribute: 'travelTime',
      defaultWeight: 15
  }
      LIMIT 3
      RETURN {
          places: p.vertices[*].label,
          travelTimes: p.edges[*].travelTime,
          travelTimeTotal: SUM(p.edges[*].travelTime)
      }
Show query results
Hide query results
[
  {
    "places": [
      "StAndrews",
      "Leuchars",
      "Edinburgh",
      "York",
      "London",
      "Brussels",
      "Cologne"
    ],
    "travelTimes": [
      0.2,
      1.5,
      3.5,
      1.8,
      2.5,
      2
    ],
    "travelTimeTotal": 11.5
  },
  {
    "places": [
      "StAndrews",
      "Leuchars",
      "Edinburgh",
      "Glasgow",
      "Carlisle",
      "Birmingham",
      "London",
      "Brussels",
      "Cologne"
    ],
    "travelTimes": [
      0.2,
      1.5,
      1,
      1,
      2,
      1.5,
      2.5,
      2
    ],
    "travelTimeTotal": 11.7
  },
  {
    "places": [
      "StAndrews",
      "Leuchars",
      "Edinburgh",
      "Glasgow",
      "Carlisle",
      "York",
      "London",
      "Brussels",
      "Cologne"
    ],
    "travelTimes": [
      0.2,
      1.5,
      1,
      1,
      2.5,
      1.8,
      2.5,
      2
    ],
    "travelTimeTotal": 12.5
  }
]

And finally clean up by removing the named graph:

arangosh> var examples = require("@arangodb/graph-examples/example-graph.js");
arangosh> examples.dropGraph("kShortestPathsGraph");
Show execution results
Hide execution results