k Paths in AQL

General query idea

This type of query finds all paths between two given documents, startVertex and targetVertex in your graph. The paths are restricted by minimum and maximum length of the paths.

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

  • an array containing the vertices on the path
  • an array containing the edges on the path

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 paths 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.

Here we have a couple of alternatives:

a) Straight way

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

b) Detour at York

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

c) Detour at Edinburgh

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

d) Detour at Edinburgh to York

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

Note that we only consider paths as valid that do not contain the same vertex twice. The following alternative would visit Aberdeen twice and will not be returned by k Paths:

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

Example Use Cases

The use-cases for k Paths are about the same as for unweighted k Shortest Paths. The main difference is that k Shortest Paths will enumerate all paths with increasing length. It will stop as soon as a given limit is reached. k Paths will instead only enumerate all paths within a given range of path length, and are thereby upper-bounded.

The k Paths traversal can be used as foundation for several other algorithms:

  • Transportation of any kind (e.g. road traffic, network package routing)
  • Flow problems: We need to transfer items from A to B, which alternatives do we have? What is their capacity?

Syntax

The syntax for k Paths queries is similar to the one for K Shortest Path with the addition to define the minimum and maximum length of the path.

It is highly recommended that you use a reasonable maximum path length or a LIMIT statement, as k Paths is a potentially expensive operation. On large connected graphs it can return a large number of paths.

Working with named graphs

FOR path
  IN MIN..MAX OUTBOUND|INBOUND|ANY K_PATHS
  startVertex TO targetVertex
  GRAPH graphName
  [OPTIONS options]
  • FOR: emits the variable path which contains one path as an object containing vertices and edges of the path.
  • IN MIN..MAX: the minimal and maximal depth for the traversal:
    • min (number, optional): paths returned by this query will have at least a length of min many edges. If not specified, it defaults to 1. The minimal possible value is 0.
    • max (number, optional): paths returned by this query will have at most a length of max many edges. If omitted, max defaults to min. Thus only the vertices and edges in the range of min are returned. max can not be specified without min.
  • OUTBOUND|INBOUND|ANY: defines in which direction edges are followed (outgoing, incoming, or both)
  • K_PATHS: the keyword to compute all 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 document identifier string or in the form of an object with the attribute _id. All other values will lead to a warning and an empty result. This is also the case if one of the specified documents does not exist.
  • 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 search. Right now there are no options that trigger an effect. However, this may change in the future.

Working with collection sets

FOR path
  IN MIN..MAX OUTBOUND|INBOUND|ANY K_PATHS
  startVertex TO targetVertex
  edgeCollection1, ..., edgeCollectionN
  [OPTIONS options]

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 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_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" : "_cuv8mmW---", 
    "label" : "Inverness" 
  }, 
  { 
    "_key" : "Aberdeen", 
    "_id" : "places/Aberdeen", 
    "_rev" : "_cuv8mmW--_", 
    "label" : "Aberdeen" 
  }, 
  { 
    "_key" : "Leuchars", 
    "_id" : "places/Leuchars", 
    "_rev" : "_cuv8mma---", 
    "label" : "Leuchars" 
  }, 
  { 
    "_key" : "StAndrews", 
    "_id" : "places/StAndrews", 
    "_rev" : "_cuv8mma--_", 
    "label" : "StAndrews" 
  }, 
  { 
    "_key" : "Edinburgh", 
    "_id" : "places/Edinburgh", 
    "_rev" : "_cuv8mme---", 
    "label" : "Edinburgh" 
  }, 
  { 
    "_key" : "Glasgow", 
    "_id" : "places/Glasgow", 
    "_rev" : "_cuv8mme--_", 
    "label" : "Glasgow" 
  }, 
  { 
    "_key" : "York", 
    "_id" : "places/York", 
    "_rev" : "_cuv8mme--A", 
    "label" : "York" 
  }, 
  { 
    "_key" : "Carlisle", 
    "_id" : "places/Carlisle", 
    "_rev" : "_cuv8mmi---", 
    "label" : "Carlisle" 
  }, 
  { 
    "_key" : "Birmingham", 
    "_id" : "places/Birmingham", 
    "_rev" : "_cuv8mmi--_", 
    "label" : "Birmingham" 
  }, 
  { 
    "_key" : "London", 
    "_id" : "places/London", 
    "_rev" : "_cuv8mmm---", 
    "label" : "London" 
  }, 
  { 
    "_key" : "Brussels", 
    "_id" : "places/Brussels", 
    "_rev" : "_cuv8mmm--_", 
    "label" : "Brussels" 
  }, 
  { 
    "_key" : "Cologne", 
    "_id" : "places/Cologne", 
    "_rev" : "_cuv8mmq---", 
    "label" : "Cologne" 
  }, 
  { 
    "_key" : "Toronto", 
    "_id" : "places/Toronto", 
    "_rev" : "_cuv8mmq--_", 
    "label" : "Toronto" 
  }, 
  { 
    "_key" : "Winnipeg", 
    "_id" : "places/Winnipeg", 
    "_rev" : "_cuv8mmq--A", 
    "label" : "Winnipeg" 
  }, 
  { 
    "_key" : "Saskatoon", 
    "_id" : "places/Saskatoon", 
    "_rev" : "_cuv8mmu---", 
    "label" : "Saskatoon" 
  }, 
  { 
    "_key" : "Edmonton", 
    "_id" : "places/Edmonton", 
    "_rev" : "_cuv8mmu--_", 
    "label" : "Edmonton" 
  }, 
  { 
    "_key" : "Jasper", 
    "_id" : "places/Jasper", 
    "_rev" : "_cuv8mmu--A", 
    "label" : "Jasper" 
  }, 
  { 
    "_key" : "Vancouver", 
    "_id" : "places/Vancouver", 
    "_rev" : "_cuv8mmu--B", 
    "label" : "Vancouver" 
  } 
]
[ 
  { 
    "_key" : "61493", 
    "_id" : "connections/61493", 
    "_from" : "places/Inverness", 
    "_to" : "places/Aberdeen", 
    "_rev" : "_cuv8mmy---", 
    "travelTime" : 3 
  }, 
  { 
    "_key" : "61495", 
    "_id" : "connections/61495", 
    "_from" : "places/Aberdeen", 
    "_to" : "places/Inverness", 
    "_rev" : "_cuv8mmy--_", 
    "travelTime" : 2.5 
  }, 
  { 
    "_key" : "61497", 
    "_id" : "connections/61497", 
    "_from" : "places/Aberdeen", 
    "_to" : "places/Leuchars", 
    "_rev" : "_cuv8mmy--A", 
    "travelTime" : 1.5 
  }, 
  { 
    "_key" : "61499", 
    "_id" : "connections/61499", 
    "_from" : "places/Leuchars", 
    "_to" : "places/Aberdeen", 
    "_rev" : "_cuv8mm2---", 
    "travelTime" : 1 
  }, 
  { 
    "_key" : "61501", 
    "_id" : "connections/61501", 
    "_from" : "places/Leuchars", 
    "_to" : "places/Edinburgh", 
    "_rev" : "_cuv8mm2--_", 
    "travelTime" : 1.5 
  }, 
  { 
    "_key" : "61503", 
    "_id" : "connections/61503", 
    "_from" : "places/Edinburgh", 
    "_to" : "places/Leuchars", 
    "_rev" : "_cuv8mm2--A", 
    "travelTime" : 3 
  }, 
  { 
    "_key" : "61505", 
    "_id" : "connections/61505", 
    "_from" : "places/Edinburgh", 
    "_to" : "places/Glasgow", 
    "_rev" : "_cuv8mm6---", 
    "travelTime" : 1 
  }, 
  { 
    "_key" : "61507", 
    "_id" : "connections/61507", 
    "_from" : "places/Glasgow", 
    "_to" : "places/Edinburgh", 
    "_rev" : "_cuv8mm6--_", 
    "travelTime" : 1 
  }, 
  { 
    "_key" : "61509", 
    "_id" : "connections/61509", 
    "_from" : "places/Edinburgh", 
    "_to" : "places/York", 
    "_rev" : "_cuv8mm6--A", 
    "travelTime" : 3.5 
  }, 
  { 
    "_key" : "61511", 
    "_id" : "connections/61511", 
    "_from" : "places/York", 
    "_to" : "places/Edinburgh", 
    "_rev" : "_cuv8mn----", 
    "travelTime" : 4 
  }, 
  { 
    "_key" : "61513", 
    "_id" : "connections/61513", 
    "_from" : "places/Glasgow", 
    "_to" : "places/Carlisle", 
    "_rev" : "_cuv8mn---_", 
    "travelTime" : 1 
  }, 
  { 
    "_key" : "61515", 
    "_id" : "connections/61515", 
    "_from" : "places/Carlisle", 
    "_to" : "places/Glasgow", 
    "_rev" : "_cuv8mn---A", 
    "travelTime" : 1 
  }, 
  { 
    "_key" : "61517", 
    "_id" : "connections/61517", 
    "_from" : "places/Carlisle", 
    "_to" : "places/York", 
    "_rev" : "_cuv8mnC---", 
    "travelTime" : 2.5 
  }, 
  { 
    "_key" : "61519", 
    "_id" : "connections/61519", 
    "_from" : "places/York", 
    "_to" : "places/Carlisle", 
    "_rev" : "_cuv8mnC--_", 
    "travelTime" : 3.5 
  }, 
  { 
    "_key" : "61521", 
    "_id" : "connections/61521", 
    "_from" : "places/Carlisle", 
    "_to" : "places/Birmingham", 
    "_rev" : "_cuv8mnG---", 
    "travelTime" : 2 
  }, 
  { 
    "_key" : "61523", 
    "_id" : "connections/61523", 
    "_from" : "places/Birmingham", 
    "_to" : "places/Carlisle", 
    "_rev" : "_cuv8mnG--_", 
    "travelTime" : 1 
  }, 
  { 
    "_key" : "61525", 
    "_id" : "connections/61525", 
    "_from" : "places/Birmingham", 
    "_to" : "places/London", 
    "_rev" : "_cuv8mnK---", 
    "travelTime" : 1.5 
  }, 
  { 
    "_key" : "61527", 
    "_id" : "connections/61527", 
    "_from" : "places/London", 
    "_to" : "places/Birmingham", 
    "_rev" : "_cuv8mnK--_", 
    "travelTime" : 2.5 
  }, 
  { 
    "_key" : "61529", 
    "_id" : "connections/61529", 
    "_from" : "places/Leuchars", 
    "_to" : "places/StAndrews", 
    "_rev" : "_cuv8mnK--A", 
    "travelTime" : 0.2 
  }, 
  { 
    "_key" : "61531", 
    "_id" : "connections/61531", 
    "_from" : "places/StAndrews", 
    "_to" : "places/Leuchars", 
    "_rev" : "_cuv8mnO---", 
    "travelTime" : 0.2 
  }, 
  { 
    "_key" : "61533", 
    "_id" : "connections/61533", 
    "_from" : "places/York", 
    "_to" : "places/London", 
    "_rev" : "_cuv8mnO--_", 
    "travelTime" : 1.8 
  }, 
  { 
    "_key" : "61535", 
    "_id" : "connections/61535", 
    "_from" : "places/London", 
    "_to" : "places/York", 
    "_rev" : "_cuv8mnS---", 
    "travelTime" : 2 
  }, 
  { 
    "_key" : "61537", 
    "_id" : "connections/61537", 
    "_from" : "places/London", 
    "_to" : "places/Brussels", 
    "_rev" : "_cuv8mnS--_", 
    "travelTime" : 2.5 
  }, 
  { 
    "_key" : "61539", 
    "_id" : "connections/61539", 
    "_from" : "places/Brussels", 
    "_to" : "places/London", 
    "_rev" : "_cuv8mnW---", 
    "travelTime" : 3.5 
  }, 
  { 
    "_key" : "61541", 
    "_id" : "connections/61541", 
    "_from" : "places/Brussels", 
    "_to" : "places/Cologne", 
    "_rev" : "_cuv8mnW--_", 
    "travelTime" : 2 
  }, 
  { 
    "_key" : "61543", 
    "_id" : "connections/61543", 
    "_from" : "places/Cologne", 
    "_to" : "places/Brussels", 
    "_rev" : "_cuv8mna---", 
    "travelTime" : 1.5 
  }, 
  { 
    "_key" : "61545", 
    "_id" : "connections/61545", 
    "_from" : "places/Toronto", 
    "_to" : "places/Winnipeg", 
    "_rev" : "_cuv8mna--_", 
    "travelTime" : 36 
  }, 
  { 
    "_key" : "61547", 
    "_id" : "connections/61547", 
    "_from" : "places/Winnipeg", 
    "_to" : "places/Toronto", 
    "_rev" : "_cuv8mna--A", 
    "travelTime" : 35 
  }, 
  { 
    "_key" : "61549", 
    "_id" : "connections/61549", 
    "_from" : "places/Winnipeg", 
    "_to" : "places/Saskatoon", 
    "_rev" : "_cuv8mna--B", 
    "travelTime" : 12 
  }, 
  { 
    "_key" : "61551", 
    "_id" : "connections/61551", 
    "_from" : "places/Saskatoon", 
    "_to" : "places/Winnipeg", 
    "_rev" : "_cuv8mne---", 
    "travelTime" : 5 
  }, 
  { 
    "_key" : "61553", 
    "_id" : "connections/61553", 
    "_from" : "places/Saskatoon", 
    "_to" : "places/Edmonton", 
    "_rev" : "_cuv8mne--_", 
    "travelTime" : 12 
  }, 
  { 
    "_key" : "61555", 
    "_id" : "connections/61555", 
    "_from" : "places/Edmonton", 
    "_to" : "places/Saskatoon", 
    "_rev" : "_cuv8mne--A", 
    "travelTime" : 17 
  }, 
  { 
    "_key" : "61557", 
    "_id" : "connections/61557", 
    "_from" : "places/Edmonton", 
    "_to" : "places/Jasper", 
    "_rev" : "_cuv8mni---", 
    "travelTime" : 6 
  }, 
  { 
    "_key" : "61559", 
    "_id" : "connections/61559", 
    "_from" : "places/Jasper", 
    "_to" : "places/Edmonton", 
    "_rev" : "_cuv8mni--_", 
    "travelTime" : 5 
  }, 
  { 
    "_key" : "61561", 
    "_id" : "connections/61561", 
    "_from" : "places/Jasper", 
    "_to" : "places/Vancouver", 
    "_rev" : "_cuv8mni--A", 
    "travelTime" : 12 
  }, 
  { 
    "_key" : "61563", 
    "_id" : "connections/61563", 
    "_from" : "places/Vancouver", 
    "_to" : "places/Jasper", 
    "_rev" : "_cuv8mnm---", 
    "travelTime" : 13 
  } 
]

Suppose we want to query all routes from Aberdeen to London.

FOR p IN 1..10 OUTBOUND K_PATHS 'places/Aberdeen' TO 'places/London'
  GRAPH 'kShortestPathsGraph'
      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
    ]
  },
  {
    "places": [
      "Aberdeen",
      "Leuchars",
      "Edinburgh",
      "Glasgow",
      "Carlisle",
      "Birmingham",
      "London"
    ],
    "travelTimes": [
      1.5,
      1.5,
      1,
      1,
      2,
      1.5
    ]
  },
  {
    "places": [
      "Aberdeen",
      "Leuchars",
      "Edinburgh",
      "Glasgow",
      "Carlisle",
      "York",
      "London"
    ],
    "travelTimes": [
      1.5,
      1.5,
      1,
      1,
      2.5,
      1.8
    ]
  },
  {
    "places": [
      "Aberdeen",
      "Leuchars",
      "Edinburgh",
      "York",
      "Carlisle",
      "Birmingham",
      "London"
    ],
    "travelTimes": [
      1.5,
      1.5,
      3.5,
      3.5,
      2,
      1.5
    ]
  }
]

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

FOR p IN 1..10 OUTBOUND K_PATHS 'places/Aberdeen' TO 'places/Toronto'
  GRAPH 'kShortestPathsGraph'
      RETURN { places: p.vertices[*].label, travelTimes: p.edges[*].travelTime }
Show query results
Hide query results
[]

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