Geo Queries

It is recommended to use AQL instead, see Geo functions.

The ArangoDB allows to select documents based on geographic coordinates. Inorder for this to work, a geo-spatial index must be defined. This index willuse a very elaborate algorithm to lookup neighbors that is a magnitude fasterthan a simple R* index.

In general a geo coordinate is a pair of latitude and longitude, which mustboth be specified as numbers. A geo index can be created on coordinates thatare stored in a single list attribute with two elements like [-10, +30] (latitude first, followed by longitude) or on coordinates stored in two separate attributes.

For example, to index the following documents, an index can be created on theposition attribute of the documents:

  1. db.test.save({ position: [ -10, 30 ] });
  2. db.test.save({ position: [ 10, 45.5 ] });
  3. db.test.ensureIndex({ type: "geo", fields: [ "position" ] });

If coordinates are stored in two distinct attributes, the index must be createdon the two attributes:

  1. db.test.save({ latitude: -10, longitude: 30 });
  2. db.test.save({ latitude: 10, longitude: 45.5 });
  3. db.test.ensureIndex({ type: "geo", fields: [ "latitude", "longitude" ] });

In order to find all documents within a given radius around a coordinate use the within operator. In order to find all documents near a given document use the near operator.

It is possible to define more than one geo-spatial index per collection. Inthis case you must give a hint using the geo operator which of indexesshould be used in a query.

Near

constructs a near query for a collectioncollection.near(latitude, longitude)

The returned list is sorted according to the distance, with the nearestdocument to the coordinate (latitude, longitude) coming first.If there are near documents of equal distance, documents are chosen randomlyfrom this set until the limit is reached. It is possible to change the limitusing the limit operator.

In order to use the near operator, a geo index must be defined for thecollection. This index also defines which attribute holds the coordinatesfor the document. If you have more then one geo-spatial index, you can usethe geo operator to select a particular index.

Note: near does not support negative skips.// However, you can still use limit followed to skip.

collection.near(latitude, longitude).limit(limit)

Limits the result to limit documents instead of the default 100.

Note: Unlike with multiple explicit limits, limit will raisethe implicit default limit imposed by within.

collection.near(latitude, longitude).distance()

This will add an attribute distance to all documents returned, whichcontains the distance between the given point and the document in meters.

collection.near(latitude, longitude).distance(name)

This will add an attribute name to all documents returned, whichcontains the distance between the given point and the document in meters.

Note: this method is not yet supported by the RocksDB storage engine.

Note: the near simple query function is deprecated as of ArangoDB 2.6.The function may be removed in future versions of ArangoDB. The preferredway for retrieving documents from a collection using the near operator isto use the AQL NEAR function in an AQL query as follows:

  1. FOR doc IN NEAR(@@collection, @latitude, @longitude, @limit)
  2. RETURN doc

Examples

To get the nearest two locations:

  1. arangosh> db.geo.ensureIndex({ type: "geo", fields: [ "loc" ] });
  2. {
  3. "bestIndexedLevel" : 17,
  4. "fields" : [
  5. "loc"
  6. ],
  7. "geoJson" : false,
  8. "id" : "geo/223",
  9. "isNewlyCreated" : true,
  10. "maxNumCoverCells" : 8,
  11. "sparse" : true,
  12. "type" : "geo",
  13. "unique" : false,
  14. "worstIndexedLevel" : 4,
  15. "code" : 201
  16. }
  17. arangosh> for (var i = -90; i <= 90; i += 10) {
  18. ........> for (var j = -180; j <= 180; j += 10) {
  19. ........> db.geo.save({
  20. ........> name : "Name/" + i + "/" + j,
  21. ........> loc: [ i, j ] });
  22. ........> } }
  23. arangosh> db.geo.near(0, 0).limit(2).toArray();
  24. [
  25. {
  26. "_key" : "927",
  27. "_id" : "geo/927",
  28. "_rev" : "_ZP4O6ba--A",
  29. "name" : "Name/0/0",
  30. "loc" : [
  31. 0,
  32. 0
  33. ]
  34. },
  35. {
  36. "_key" : "1001",
  37. "_id" : "geo/1001",
  38. "_rev" : "_ZP4O6cW--A",
  39. "name" : "Name/10/0",
  40. "loc" : [
  41. 10,
  42. 0
  43. ]
  44. }
  45. ]

Hide execution results

  1. arangosh> db.geo.ensureIndex({ type: "geo", fields: [ "loc" ] });
  2. arangosh> for (var i = -90; i <= 90; i += 10) {
  3. ........> for (var j = -180; j <= 180; j += 10) {
  4. ........> db.geo.save({
  5. ........> name : "Name/" + i + "/" + j,
  6. ........> loc: [ i, j ] });
  7. ........> } }
  8. arangosh> db.geo.near(0, 0).limit(2).toArray();

Show execution results

If you need the distance as well, then you can use the distanceoperator:

  1. arangosh> db.geo.ensureIndex({ type: "geo", fields: [ "loc" ] });
  2. {
  3. "bestIndexedLevel" : 17,
  4. "fields" : [
  5. "loc"
  6. ],
  7. "geoJson" : false,
  8. "id" : "geo/1643",
  9. "isNewlyCreated" : true,
  10. "maxNumCoverCells" : 8,
  11. "sparse" : true,
  12. "type" : "geo",
  13. "unique" : false,
  14. "worstIndexedLevel" : 4,
  15. "code" : 201
  16. }
  17. arangosh> for (var i = -90; i <= 90; i += 10) {
  18. ........> for (var j = -180; j <= 180; j += 10) {
  19. ........> db.geo.save({
  20. ........> name : "Name/" + i + "/" + j,
  21. ........> loc: [ i, j ] });
  22. ........> } }
  23. arangosh> db.geo.near(0, 0).distance().limit(2).toArray();
  24. [
  25. {
  26. "_id" : "geo/2347",
  27. "_key" : "2347",
  28. "_rev" : "_ZP4O6u---D",
  29. "loc" : [
  30. 0,
  31. 0
  32. ],
  33. "name" : "Name/0/0",
  34. "distance" : 0
  35. },
  36. {
  37. "_id" : "geo/2421",
  38. "_key" : "2421",
  39. "_rev" : "_ZP4O6u2--A",
  40. "loc" : [
  41. 10,
  42. 0
  43. ],
  44. "name" : "Name/10/0",
  45. "distance" : 1111949.2664455872
  46. }
  47. ]

Hide execution results

  1. arangosh> db.geo.ensureIndex({ type: "geo", fields: [ "loc" ] });
  2. arangosh> for (var i = -90; i <= 90; i += 10) {
  3. ........> for (var j = -180; j <= 180; j += 10) {
  4. ........> db.geo.save({
  5. ........> name : "Name/" + i + "/" + j,
  6. ........> loc: [ i, j ] });
  7. ........> } }
  8. arangosh> db.geo.near(0, 0).distance().limit(2).toArray();

Show execution results

Within

constructs a within query for a collectioncollection.within(latitude, longitude, radius)

This will find all documents within a given radius around the coordinate(latitude, longitude). The returned array is sorted by distance,beginning with the nearest document.

In order to use the within operator, a geo index must be defined for thecollection. This index also defines which attribute holds the coordinatesfor the document. If you have more then one geo-spatial index, you can usethe geo operator to select a particular index.

collection.within(latitude, longitude, radius).distance()

This will add an attribute _distance to all documents returned, whichcontains the distance between the given point and the document in meters.

collection.within(latitude, longitude, radius).distance(name)

This will add an attribute name to all documents returned, whichcontains the distance between the given point and the document in meters.

Note: this method is not yet supported by the RocksDB storage engine.

Note: the within simple query function is deprecated as of ArangoDB 2.6.The function may be removed in future versions of ArangoDB. The preferredway for retrieving documents from a collection using the within operator isto use the AQL WITHIN function in an AQL query as follows:

  1. FOR doc IN WITHIN(@@collection, @latitude, @longitude, @radius, @distanceAttributeName)
  2. RETURN doc

Examples

To find all documents within a radius of 2000 km use:

  1. arangosh> for (var i = -90; i <= 90; i += 10) {
  2. ........> for (var j = -180; j <= 180; j += 10) {
  3. ........> db.geo.save({ name : "Name/" + i + "/" + j, loc: [ i, j ] }); } }
  4. arangosh> db.geo.within(0, 0, 2000 * 1000).distance().toArray();
  5. [
  6. {
  7. "_id" : "geo/3767",
  8. "_key" : "3767",
  9. "_rev" : "_ZP4O7Ai--A",
  10. "loc" : [
  11. 0,
  12. 0
  13. ],
  14. "name" : "Name/0/0",
  15. "distance" : 0
  16. },
  17. {
  18. "_id" : "geo/3841",
  19. "_key" : "3841",
  20. "_rev" : "_ZP4O7Ba--A",
  21. "loc" : [
  22. 10,
  23. 0
  24. ],
  25. "name" : "Name/10/0",
  26. "distance" : 1111949.2664455872
  27. },
  28. {
  29. "_id" : "geo/3769",
  30. "_key" : "3769",
  31. "_rev" : "_ZP4O7Ai--C",
  32. "loc" : [
  33. 0,
  34. 10
  35. ],
  36. "name" : "Name/0/10",
  37. "distance" : 1111949.2664455872
  38. },
  39. {
  40. "_id" : "geo/3693",
  41. "_key" : "3693",
  42. "_rev" : "_ZP4O7_u---",
  43. "loc" : [
  44. -10,
  45. 0
  46. ],
  47. "name" : "Name/-10/0",
  48. "distance" : 1111949.2664455872
  49. },
  50. {
  51. "_id" : "geo/3765",
  52. "_key" : "3765",
  53. "_rev" : "_ZP4O7Ai---",
  54. "loc" : [
  55. 0,
  56. -10
  57. ],
  58. "name" : "Name/0/-10",
  59. "distance" : 1111949.2664455872
  60. },
  61. {
  62. "_id" : "geo/3691",
  63. "_key" : "3691",
  64. "_rev" : "_ZP4O7_q--A",
  65. "loc" : [
  66. -10,
  67. -10
  68. ],
  69. "name" : "Name/-10/-10",
  70. "distance" : 1568520.556798576
  71. },
  72. {
  73. "_id" : "geo/3843",
  74. "_key" : "3843",
  75. "_rev" : "_ZP4O7Ba--C",
  76. "loc" : [
  77. 10,
  78. 10
  79. ],
  80. "name" : "Name/10/10",
  81. "distance" : 1568520.556798576
  82. },
  83. {
  84. "_id" : "geo/3839",
  85. "_key" : "3839",
  86. "_rev" : "_ZP4O7Ba---",
  87. "loc" : [
  88. 10,
  89. -10
  90. ],
  91. "name" : "Name/10/-10",
  92. "distance" : 1568520.556798576
  93. },
  94. {
  95. "_id" : "geo/3695",
  96. "_key" : "3695",
  97. "_rev" : "_ZP4O7_u--A",
  98. "loc" : [
  99. -10,
  100. 10
  101. ],
  102. "name" : "Name/-10/10",
  103. "distance" : 1568520.556798576
  104. }
  105. ]

Hide execution results

  1. arangosh> for (var i = -90; i <= 90; i += 10) {
  2. ........> for (var j = -180; j <= 180; j += 10) {
  3. ........> db.geo.save({ name : "Name/" + i + "/" + j, loc: [ i, j ] }); } }
  4. arangosh> db.geo.within(0, 0, 2000 * 1000).distance().toArray();

Show execution results

Geo

constructs a geo index selectioncollection.geo(location-attribute)

Looks up a geo index defined on attribute location_attribute.

Returns a geo index object if an index was found. The near orwithin operators can then be used to execute a geo-spatial query onthis particular index.

This is useful for collections with multiple defined geo indexes.

collection.geo(location_attribute, true)

Looks up a geo index on a compound attribute location_attribute.

Returns a geo index object if an index was found. The near orwithin operators can then be used to execute a geo-spatial query onthis particular index.

collection.geo(latitude_attribute, longitude_attribute)

Looks up a geo index defined on the two attributes latitude_attribute_and _longitude-attribute.

Returns a geo index object if an index was found. The near orwithin operators can then be used to execute a geo-spatial query onthis particular index.

Note: this method is not yet supported by the RocksDB storage engine.

Note: the geo simple query helper function is deprecated as of ArangoDB2.6. The function may be removed in future versions of ArangoDB. The preferredway for running geo queries is to use their AQL equivalents.

Examples

Assume you have a location stored as list in the attribute home_and a destination stored in the attribute _work. Then you can use thegeo operator to select which geo-spatial attributes (and thus whichindex) to use in a near query.

  1. arangosh> for (i = -90; i <= 90; i += 10) {
  2. ........> for (j = -180; j <= 180; j += 10) {
  3. ........> db.complex.save({ name : "Name/" + i + "/" + j,
  4. ........> home : [ i, j ],
  5. ........> work : [ -i, -j ] });
  6. ........> }
  7. ........> }
  8. ........>
  9. arangosh> db.complex.near(0, 170).limit(5);
  10. [ArangoError 1570: no suitable geo index found for geo restriction on 'complex']
  11. arangosh> db.complex.ensureIndex({ type: "geo", fields: [ "home" ] });
  12. {
  13. "bestIndexedLevel" : 17,
  14. "fields" : [
  15. "home"
  16. ],
  17. "geoJson" : false,
  18. "id" : "complex/85111",
  19. "isNewlyCreated" : true,
  20. "maxNumCoverCells" : 8,
  21. "sparse" : true,
  22. "type" : "geo",
  23. "unique" : false,
  24. "worstIndexedLevel" : 4,
  25. "code" : 201
  26. }
  27. arangosh> db.complex.near(0, 170).limit(5).toArray();
  28. [
  29. {
  30. "_key" : "84439",
  31. "_id" : "complex/84439",
  32. "_rev" : "_ZP4PX0y--A",
  33. "name" : "Name/0/170",
  34. "home" : [
  35. 0,
  36. 170
  37. ],
  38. "work" : [
  39. 0,
  40. -170
  41. ]
  42. },
  43. {
  44. "_key" : "84441",
  45. "_id" : "complex/84441",
  46. "_rev" : "_ZP4PX0y--C",
  47. "name" : "Name/0/180",
  48. "home" : [
  49. 0,
  50. 180
  51. ],
  52. "work" : [
  53. 0,
  54. -180
  55. ]
  56. },
  57. {
  58. "_key" : "84513",
  59. "_id" : "complex/84513",
  60. "_rev" : "_ZP4PX1q--A",
  61. "name" : "Name/10/170",
  62. "home" : [
  63. 10,
  64. 170
  65. ],
  66. "work" : [
  67. -10,
  68. -170
  69. ]
  70. },
  71. {
  72. "_key" : "84365",
  73. "_id" : "complex/84365",
  74. "_rev" : "_ZP4PXz2--C",
  75. "name" : "Name/-10/170",
  76. "home" : [
  77. -10,
  78. 170
  79. ],
  80. "work" : [
  81. 10,
  82. -170
  83. ]
  84. },
  85. {
  86. "_key" : "84369",
  87. "_id" : "complex/84369",
  88. "_rev" : "_ZP4PXz6--A",
  89. "name" : "Name/0/-180",
  90. "home" : [
  91. 0,
  92. -180
  93. ],
  94. "work" : [
  95. 0,
  96. 180
  97. ]
  98. }
  99. ]
  100. arangosh> db.complex.geo("work").near(0, 170).limit(5);
  101. [ArangoError 1570: no suitable geo index found for geo restriction on 'complex']
  102. arangosh> db.complex.ensureIndex({ type: "geo", fields: [ "work" ] });
  103. {
  104. "bestIndexedLevel" : 17,
  105. "fields" : [
  106. "work"
  107. ],
  108. "geoJson" : false,
  109. "id" : "complex/85119",
  110. "isNewlyCreated" : true,
  111. "maxNumCoverCells" : 8,
  112. "sparse" : true,
  113. "type" : "geo",
  114. "unique" : false,
  115. "worstIndexedLevel" : 4,
  116. "code" : 201
  117. }
  118. arangosh> db.complex.geo("work").near(0, 170).limit(5).toArray();
  119. [
  120. {
  121. "_key" : "84439",
  122. "_id" : "complex/84439",
  123. "_rev" : "_ZP4PX0y--A",
  124. "name" : "Name/0/170",
  125. "home" : [
  126. 0,
  127. 170
  128. ],
  129. "work" : [
  130. 0,
  131. -170
  132. ]
  133. },
  134. {
  135. "_key" : "84441",
  136. "_id" : "complex/84441",
  137. "_rev" : "_ZP4PX0y--C",
  138. "name" : "Name/0/180",
  139. "home" : [
  140. 0,
  141. 180
  142. ],
  143. "work" : [
  144. 0,
  145. -180
  146. ]
  147. },
  148. {
  149. "_key" : "84513",
  150. "_id" : "complex/84513",
  151. "_rev" : "_ZP4PX1q--A",
  152. "name" : "Name/10/170",
  153. "home" : [
  154. 10,
  155. 170
  156. ],
  157. "work" : [
  158. -10,
  159. -170
  160. ]
  161. },
  162. {
  163. "_key" : "84365",
  164. "_id" : "complex/84365",
  165. "_rev" : "_ZP4PXz2--C",
  166. "name" : "Name/-10/170",
  167. "home" : [
  168. -10,
  169. 170
  170. ],
  171. "work" : [
  172. 10,
  173. -170
  174. ]
  175. },
  176. {
  177. "_key" : "84369",
  178. "_id" : "complex/84369",
  179. "_rev" : "_ZP4PXz6--A",
  180. "name" : "Name/0/-180",
  181. "home" : [
  182. 0,
  183. -180
  184. ],
  185. "work" : [
  186. 0,
  187. 180
  188. ]
  189. }
  190. ]

Hide execution results

  1. arangosh> for (i = -90; i <= 90; i += 10) {
  2. ........> for (j = -180; j <= 180; j += 10) {
  3. ........> db.complex.save({ name : "Name/" + i + "/" + j,
  4. ........> home : [ i, j ],
  5. ........> work : [ -i, -j ] });
  6. ........> }
  7. ........> }
  8. ........>
  9. arangosh> db.complex.near(0, 170).limit(5);
  10. arangosh> db.complex.ensureIndex({ type: "geo", fields: [ "home" ] });
  11. arangosh> db.complex.near(0, 170).limit(5).toArray();
  12. arangosh> db.complex.geo("work").near(0, 170).limit(5);
  13. arangosh> db.complex.ensureIndex({ type: "geo", fields: [ "work" ] });
  14. arangosh> db.complex.geo("work").near(0, 170).limit(5).toArray();

Show execution results

Other ArangoDB geographic features are described in: