Elasticsearch сопоставляет ключевое слово tokenizer, чтобы избежать разделения токенов и разрешить использование подстановочных знаков

Я пытаюсь сделать функцию автозаполнения с угловыми и elasticsearch в заданном поле, например countryname. он может содержать простые имена, такие как "Франция", "Испания" или "составленные имена", такие как "Сьерра-Леоне".

В отображении это поле not_analyzed, чтобы предотвратить эластичность, чтобы токенизировать "сложенные имена"

"COUNTRYNAME" : {"type" : "string", "store" : "yes","index": "not_analyzed" }

Мне нужно запросить elasticsearch:

  • чтобы отфильтровать документ с чем-то вроде "countryname: value", где значение может содержать подстановочный знак
  • и сделать агрегацию по имени страны, возвращенному фильтром (я делаю агрегацию, чтобы получать только отдельные данные, счет для меня бесполезен, возможно, есть лучшее решение)

Я не могу использовать подстановочный знак с полем "not_analyzed":

это мой запрос, но подстановочный знак в переменной "значение" не работает и чувствителен к регистру:

Только подстановочный знак ее работы:

curl -XGET 'local_host:9200/botanic/specimens/_search?size=0' -d '{
  "fields": [
    "COUNTRYNAME"
  ],
  "query": {
    "query_string": {
      "query": "COUNTRYNAME:*"
    }
  },
  "aggs": {
    "general": {
      "terms": {
        "field": "COUNTRYNAME",
        "size": 0
      }
    }
  }
}'

но это не работает (franc *):

curl -XGET 'local_host:9200/botanic/specimens/_search?size=0' -d '{
  "fields": [
    "COUNTRYNAME"
  ],
  "query": {
    "query_string": {
      "query": "COUNTRYNAME:Franc*"
    }
  },
  "aggs": {
    "general": {
      "terms": {
        "field": "COUNTRYNAME",
        "size": 0
      }
    }
  }
}'

Я также пробовал с bool must query, но не работаю с этим не_аналитическим полем и подстановочным знаком:

curl -XGET 'local_host:9200/botanic/specimens/_search?size=0' -d '{
  "fields": [
    "COUNTRYNAME"
  ],
  "query": {
    "bool": {
      "must": [
        {
          "match": {
            "COUNTRYNAME": "Franc*"
          }
        }
      ]
    }
  },
  "aggs": {
    "general": {
      "terms": {
        "field": "COUNTRYNAME",
        "size": 0
      }
    }
  }
}'

Что мне не хватает или что-то не так? должен ли я оставить поле analyzed в отображении и использовать другой анализатор, который не разбивает скомпонованное имя на токен??

Ответ 1

я нашел рабочее решение: токенизатор ключевого слова. создать собственный анализатор и использовать его в сопоставлении для поля, которое я хочу сохранить без разделения по пространству:

    curl -XPUT 'localhost:9200/botanic/' -d '{
 "settings":{
     "index":{
        "analysis":{
           "analyzer":{
              "keylower":{
                 "tokenizer":"keyword",
                 "filter":"lowercase"
              }
           }
        }
     }
  },
  "mappings":{
        "specimens" : {
            "_all" : {"enabled" : true},
            "_index" : {"enabled" : true},
            "_id" : {"index": "not_analyzed", "store" : false},
            "properties" : {
                "_id" : {"type" : "string", "store" : "no","index": "not_analyzed"  } ,
            ...
                "LOCATIONID" : {"type" : "string",  "store" : "yes","index": "not_analyzed" } ,
                "AVERAGEALTITUDEROUNDED" : {"type" : "string",  "store" : "yes","index": "analyzed" } ,
                "CONTINENT" : {"type" : "string","analyzer":"keylower" } ,
                "COUNTRYNAME" : {"type" : "string","analyzer":"keylower" } ,                
                "COUNTRYCODE" : {"type" : "string", "store" : "yes","index": "analyzed" } ,
                "COUNTY" : {"type" : "string","analyzer":"keylower" } ,
                "LOCALITY" : {"type" : "string","analyzer":"keylower" }                 
            }
        }
    }
}'

поэтому я могу использовать подстановочный знак в запросе в поле COUNTRYNAME, которое не разделено:

curl -XGET 'localhost:9200/botanic/specimens/_search?size=10' -d '{
"fields"  : ["COUNTRYNAME"],     
"query": {"query_string" : {
                    "query": "COUNTRYNAME:bol*"
}},
"aggs" : {
    "general" : {
        "terms" : {
            "field" : "COUNTRYNAME", "size":0
        }
    }
}}'

результат:

{
    "took" : 14,
    "timed_out" : false,
    "_shards" : {
        "total" : 5,
        "successful" : 5,
        "failed" : 0
    },
    "hits" : {
        "total" : 45,
        "max_score" : 1.0,
        "hits" : [{
                "_index" : "botanic",
                "_type" : "specimens",
                "_id" : "91E7B53B61DF4E76BF70C780315A5DFD",
                "_score" : 1.0,
                "fields" : {
                    "COUNTRYNAME" : ["Bolivia, Plurinational State of"]
                }
            }, {
                "_index" : "botanic",
                "_type" : "specimens",
                "_id" : "7D811B5D08FF4F17BA174A3D294B5986",
                "_score" : 1.0,
                "fields" : {
                    "COUNTRYNAME" : ["Bolivia, Plurinational State of"]
                }
            } ...
        ]
    },
    "aggregations" : {
        "general" : {
            "buckets" : [{
                    "key" : "bolivia, plurinational state of",
                    "doc_count" : 45
                }
            ]
        }
    }
}