web-dev-qa-db-fra.com

Distance de retour dans les résultats de recherche élastique?

Ma question est similaire à ceci n .

Existe-t-il simplement un moyen de renvoyer la distance géographique lorsque le tri n'est PAS effectué avec _geo_distance?

Mise à jour: pour clarifier, je veux les résultats dans un ordre aléatoire ET inclure la distance.

36
Yeggeps

Oui, vous pouvez, en utilisant un champ de script .

Par exemple, en supposant que votre document possède un champ de géo-point appelé location, vous pouvez utiliser ce qui suit:

(noter la \u0027 n'est qu'un guillemet simple échappé, donc \u0027location\u0027 est vraiment 'location')

curl -XGET 'http://127.0.0.1:9200/geonames/_search?pretty=1'  -d '
{
   "script_fields" : {
      "distance" : {
         "params" : {
            "lat" : 2.27,
            "lon" : 50.3
         },
         "script" : "doc[\u0027location\u0027].distanceInKm(lat,lon)"
      }
   }
}
'

# [Thu Feb 16 11:20:29 2012] Response:
# {
#    "hits" : {
#       "hits" : [
#          {
#             "_score" : 1,
#             "fields" : {
#                "distance" : 466.844095463887
#             },
#             "_index" : "geonames_1318324623",
#             "_id" : "6436641_en",
#             "_type" : "place"
#          },
... etc

Si vous voulez que le _source champ à renvoyer également, vous pouvez alors le spécifier comme suit:

curl -XGET 'http://127.0.0.1:9200/geonames/_search?pretty=1'  -d '
{
   "fields" : [ "_source" ],
   "script_fields" : {
      "distance" : {
         "params" : {
            "lat" : 2.27,
            "lon" : 50.3
         },
         "script" : "doc[\u0027location\u0027].distanceInKm(lat,lon)"
      }
   }
}
'
44
DrTech

Excellente réponse de DrTech ... voici une version mise à jour pour Elasticsearch 5.x avec indolore comme langage de script. J'ai également ajouté "store_fields" pour inclure _source dans le résultat:

curl -XGET 'http://127.0.0.1:9200/geonames/_search?pretty=1'  -d '
{
  "stored_fields" : [ "_source" ],
  "script_fields" : {
    "distance" : {
      "script" : {
        "inline": "doc['location'].arcDistance(params.lat,params.lon) * 0.001",
        "lang": "painless",
        "params": {
          "lat": 2.27,
          "lon": 50.3
        }
      }
    }
  }
}'
12
Jette

Pour renvoyer la distance comme tous les champs/sources par défaut, vous pouvez également procéder comme suit:

Pour éviter qu'il trie par distance (principalement), vous triez simplement par _score (ou par ce que vous voulez que les résultats soient triés) en premier.

{
   "sort": [
    "_score",
    {
      "_geo_distance": {
        "location": { 
          "lat":  40.715,
          "lon": -73.998
        },
        "order":         "asc",
        "unit":          "km", 
        "distance_type": "plane" 
      }
    }
  ]
}
8

Puisque ES 1. MVEL est désactivé par défaut, utilisez donc une requête comme:

GET some-index/_search
{
  "sort": [
    {
      "_geo_distance": {
        "geo_location": "47.1, 8.1",
        "order": "asc",
        "unit": "m"
      }
    }
  ],
  "query": {
    "match_all": {}
  },
   "script_fields" : {
      "distance" : {
         "lang": "groovy",
         "params" : {
            "lat" : 47.1,
            "lon" : 8.1
         },
         "script" : "doc[\u0027geo_location\u0027].distanceInKm(lat,lon)"
      }
   }
}

voir: "lang": "groovy", partie

3
user1266222