2013-08-28 19:24:34 -04:00
|
|
|
[[analysis-edgengram-tokenizer]]
|
2019-11-22 10:38:01 -05:00
|
|
|
=== Edge n-gram tokenizer
|
2013-08-28 19:24:34 -04:00
|
|
|
|
2016-05-19 13:42:23 -04:00
|
|
|
The `edge_ngram` tokenizer first breaks text down into words whenever it
|
|
|
|
encounters one of a list of specified characters, then it emits
|
|
|
|
https://en.wikipedia.org/wiki/N-gram[N-grams] of each word where the start of
|
|
|
|
the N-gram is anchored to the beginning of the word.
|
2013-08-28 19:24:34 -04:00
|
|
|
|
2016-05-19 13:42:23 -04:00
|
|
|
Edge N-Grams are useful for _search-as-you-type_ queries.
|
2013-08-28 19:24:34 -04:00
|
|
|
|
2016-05-19 13:42:23 -04:00
|
|
|
TIP: When you need _search-as-you-type_ for text which has a widely known
|
|
|
|
order, such as movie or song titles, the
|
2019-07-19 09:16:35 -04:00
|
|
|
<<completion-suggester,completion suggester>> is a much more efficient
|
2016-05-19 13:42:23 -04:00
|
|
|
choice than edge N-grams. Edge N-grams have the advantage when trying to
|
|
|
|
autocomplete words that can appear in any order.
|
|
|
|
|
|
|
|
[float]
|
|
|
|
=== Example output
|
|
|
|
|
|
|
|
With the default settings, the `edge_ngram` tokenizer treats the initial text as a
|
|
|
|
single token and produces N-grams with minimum length `1` and maximum length
|
|
|
|
`2`:
|
|
|
|
|
2019-09-09 13:38:14 -04:00
|
|
|
[source,console]
|
2016-05-19 13:42:23 -04:00
|
|
|
---------------------------
|
|
|
|
POST _analyze
|
|
|
|
{
|
|
|
|
"tokenizer": "edge_ngram",
|
|
|
|
"text": "Quick Fox"
|
|
|
|
}
|
|
|
|
---------------------------
|
|
|
|
|
|
|
|
/////////////////////
|
|
|
|
|
2019-09-06 09:22:08 -04:00
|
|
|
[source,console-result]
|
2016-05-19 13:42:23 -04:00
|
|
|
----------------------------
|
|
|
|
{
|
|
|
|
"tokens": [
|
|
|
|
{
|
|
|
|
"token": "Q",
|
|
|
|
"start_offset": 0,
|
|
|
|
"end_offset": 1,
|
|
|
|
"type": "word",
|
|
|
|
"position": 0
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"token": "Qu",
|
|
|
|
"start_offset": 0,
|
|
|
|
"end_offset": 2,
|
|
|
|
"type": "word",
|
|
|
|
"position": 1
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
----------------------------
|
|
|
|
|
|
|
|
/////////////////////
|
2013-08-28 19:24:34 -04:00
|
|
|
|
|
|
|
|
2016-05-19 13:42:23 -04:00
|
|
|
The above sentence would produce the following terms:
|
2013-08-28 19:24:34 -04:00
|
|
|
|
2016-05-19 13:42:23 -04:00
|
|
|
[source,text]
|
|
|
|
---------------------------
|
|
|
|
[ Q, Qu ]
|
|
|
|
---------------------------
|
2013-08-28 19:24:34 -04:00
|
|
|
|
2016-05-19 13:42:23 -04:00
|
|
|
NOTE: These default gram lengths are almost entirely useless. You need to
|
|
|
|
configure the `edge_ngram` before using it.
|
2013-08-28 19:24:34 -04:00
|
|
|
|
2016-05-19 13:42:23 -04:00
|
|
|
[float]
|
|
|
|
=== Configuration
|
|
|
|
|
|
|
|
The `edge_ngram` tokenizer accepts the following parameters:
|
2013-08-28 19:24:34 -04:00
|
|
|
|
2016-05-19 13:42:23 -04:00
|
|
|
`min_gram`::
|
|
|
|
Minimum length of characters in a gram. Defaults to `1`.
|
|
|
|
|
|
|
|
`max_gram`::
|
2019-11-13 14:27:10 -05:00
|
|
|
+
|
|
|
|
--
|
|
|
|
Maximum length of characters in a gram. Defaults to `2`.
|
|
|
|
|
|
|
|
See <<max-gram-limits>>.
|
|
|
|
--
|
2016-05-19 13:42:23 -04:00
|
|
|
|
|
|
|
`token_chars`::
|
|
|
|
|
|
|
|
Character classes that should be included in a token. Elasticsearch
|
|
|
|
will split on characters that don't belong to the classes specified.
|
|
|
|
Defaults to `[]` (keep all characters).
|
|
|
|
+
|
|
|
|
Character classes may be any of the following:
|
|
|
|
+
|
|
|
|
* `letter` -- for example `a`, `b`, `ï` or `京`
|
|
|
|
* `digit` -- for example `3` or `7`
|
|
|
|
* `whitespace` -- for example `" "` or `"\n"`
|
|
|
|
* `punctuation` -- for example `!` or `"`
|
|
|
|
* `symbol` -- for example `$` or `√`
|
2019-11-20 04:36:39 -05:00
|
|
|
* `custom` -- custom characters which need to be set using the
|
|
|
|
`custom_token_chars` setting.
|
|
|
|
|
|
|
|
`custom_token_chars`::
|
|
|
|
|
|
|
|
Custom characters that should be treated as part of a token. For example,
|
|
|
|
setting this to `+-_` will make the tokenizer treat the plus, minus and
|
|
|
|
underscore sign as part of a token.
|
2013-08-28 19:24:34 -04:00
|
|
|
|
2019-11-13 14:27:10 -05:00
|
|
|
[[max-gram-limits]]
|
|
|
|
=== Limitations of the `max_gram` parameter
|
|
|
|
|
|
|
|
The `edge_ngram` tokenizer's `max_gram` value limits the character length of
|
|
|
|
tokens. When the `edge_ngram` tokenizer is used with an index analyzer, this
|
|
|
|
means search terms longer than the `max_gram` length may not match any indexed
|
|
|
|
terms.
|
|
|
|
|
|
|
|
For example, if the `max_gram` is `3`, searches for `apple` won't match the
|
|
|
|
indexed term `app`.
|
|
|
|
|
2019-11-22 10:38:01 -05:00
|
|
|
To account for this, you can use the
|
|
|
|
<<analysis-truncate-tokenfilter,`truncate`>> token filter with a search analyzer
|
|
|
|
to shorten search terms to the `max_gram` character length. However, this could
|
|
|
|
return irrelevant results.
|
2019-11-13 14:27:10 -05:00
|
|
|
|
|
|
|
For example, if the `max_gram` is `3` and search terms are truncated to three
|
|
|
|
characters, the search term `apple` is shortened to `app`. This means searches
|
|
|
|
for `apple` return any indexed terms matching `app`, such as `apply`, `snapped`,
|
|
|
|
and `apple`.
|
|
|
|
|
|
|
|
We recommend testing both approaches to see which best fits your
|
|
|
|
use case and desired search experience.
|
|
|
|
|
2013-08-28 19:24:34 -04:00
|
|
|
[float]
|
2016-05-19 13:42:23 -04:00
|
|
|
=== Example configuration
|
|
|
|
|
|
|
|
In this example, we configure the `edge_ngram` tokenizer to treat letters and
|
|
|
|
digits as tokens, and to produce grams with minimum length `2` and maximum
|
|
|
|
length `10`:
|
|
|
|
|
2019-09-09 13:38:14 -04:00
|
|
|
[source,console]
|
2016-05-19 13:42:23 -04:00
|
|
|
----------------------------
|
2019-01-22 09:13:52 -05:00
|
|
|
PUT my_index
|
2016-05-19 13:42:23 -04:00
|
|
|
{
|
|
|
|
"settings": {
|
|
|
|
"analysis": {
|
|
|
|
"analyzer": {
|
|
|
|
"my_analyzer": {
|
|
|
|
"tokenizer": "my_tokenizer"
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"tokenizer": {
|
|
|
|
"my_tokenizer": {
|
|
|
|
"type": "edge_ngram",
|
|
|
|
"min_gram": 2,
|
|
|
|
"max_gram": 10,
|
|
|
|
"token_chars": [
|
|
|
|
"letter",
|
|
|
|
"digit"
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
POST my_index/_analyze
|
|
|
|
{
|
|
|
|
"analyzer": "my_analyzer",
|
|
|
|
"text": "2 Quick Foxes."
|
|
|
|
}
|
|
|
|
----------------------------
|
|
|
|
|
|
|
|
/////////////////////
|
2013-08-28 19:24:34 -04:00
|
|
|
|
2019-09-06 09:22:08 -04:00
|
|
|
[source,console-result]
|
2016-05-19 13:42:23 -04:00
|
|
|
----------------------------
|
|
|
|
{
|
|
|
|
"tokens": [
|
|
|
|
{
|
|
|
|
"token": "Qu",
|
|
|
|
"start_offset": 2,
|
|
|
|
"end_offset": 4,
|
|
|
|
"type": "word",
|
|
|
|
"position": 0
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"token": "Qui",
|
|
|
|
"start_offset": 2,
|
|
|
|
"end_offset": 5,
|
|
|
|
"type": "word",
|
|
|
|
"position": 1
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"token": "Quic",
|
|
|
|
"start_offset": 2,
|
|
|
|
"end_offset": 6,
|
|
|
|
"type": "word",
|
|
|
|
"position": 2
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"token": "Quick",
|
|
|
|
"start_offset": 2,
|
|
|
|
"end_offset": 7,
|
|
|
|
"type": "word",
|
|
|
|
"position": 3
|
|
|
|
},
|
2013-08-28 19:24:34 -04:00
|
|
|
{
|
2016-05-19 13:42:23 -04:00
|
|
|
"token": "Fo",
|
|
|
|
"start_offset": 8,
|
|
|
|
"end_offset": 10,
|
|
|
|
"type": "word",
|
|
|
|
"position": 4
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"token": "Fox",
|
|
|
|
"start_offset": 8,
|
|
|
|
"end_offset": 11,
|
|
|
|
"type": "word",
|
|
|
|
"position": 5
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"token": "Foxe",
|
|
|
|
"start_offset": 8,
|
|
|
|
"end_offset": 12,
|
|
|
|
"type": "word",
|
|
|
|
"position": 6
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"token": "Foxes",
|
|
|
|
"start_offset": 8,
|
|
|
|
"end_offset": 13,
|
|
|
|
"type": "word",
|
|
|
|
"position": 7
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
----------------------------
|
|
|
|
|
|
|
|
/////////////////////
|
|
|
|
|
|
|
|
The above example produces the following terms:
|
|
|
|
|
|
|
|
[source,text]
|
|
|
|
---------------------------
|
|
|
|
[ Qu, Qui, Quic, Quick, Fo, Fox, Foxe, Foxes ]
|
|
|
|
---------------------------
|
|
|
|
|
|
|
|
Usually we recommend using the same `analyzer` at index time and at search
|
2019-11-13 14:27:10 -05:00
|
|
|
time. In the case of the `edge_ngram` tokenizer, the advice is different. It
|
2016-05-19 13:42:23 -04:00
|
|
|
only makes sense to use the `edge_ngram` tokenizer at index time, to ensure
|
2019-11-13 14:27:10 -05:00
|
|
|
that partial words are available for matching in the index. At search time,
|
2016-05-19 13:42:23 -04:00
|
|
|
just search for the terms the user has typed in, for instance: `Quick Fo`.
|
|
|
|
|
2019-11-13 14:27:10 -05:00
|
|
|
Below is an example of how to set up a field for _search-as-you-type_.
|
|
|
|
|
|
|
|
Note that the `max_gram` value for the index analyzer is `10`, which limits
|
|
|
|
indexed terms to 10 characters. Search terms are not truncated, meaning that
|
|
|
|
search terms longer than 10 characters may not match any indexed terms.
|
2016-05-19 13:42:23 -04:00
|
|
|
|
2019-09-09 13:38:14 -04:00
|
|
|
[source,console]
|
2016-05-19 13:42:23 -04:00
|
|
|
-----------------------------------
|
2019-01-22 09:13:52 -05:00
|
|
|
PUT my_index
|
2016-05-19 13:42:23 -04:00
|
|
|
{
|
|
|
|
"settings": {
|
|
|
|
"analysis": {
|
|
|
|
"analyzer": {
|
|
|
|
"autocomplete": {
|
|
|
|
"tokenizer": "autocomplete",
|
|
|
|
"filter": [
|
|
|
|
"lowercase"
|
|
|
|
]
|
|
|
|
},
|
|
|
|
"autocomplete_search": {
|
|
|
|
"tokenizer": "lowercase"
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"tokenizer": {
|
|
|
|
"autocomplete": {
|
|
|
|
"type": "edge_ngram",
|
|
|
|
"min_gram": 2,
|
|
|
|
"max_gram": 10,
|
|
|
|
"token_chars": [
|
|
|
|
"letter"
|
|
|
|
]
|
2013-08-28 19:24:34 -04:00
|
|
|
}
|
2016-05-19 13:42:23 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"mappings": {
|
2019-01-22 09:13:52 -05:00
|
|
|
"properties": {
|
|
|
|
"title": {
|
|
|
|
"type": "text",
|
|
|
|
"analyzer": "autocomplete",
|
|
|
|
"search_analyzer": "autocomplete_search"
|
2016-05-19 13:42:23 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-08-28 19:24:34 -04:00
|
|
|
|
2017-12-14 11:47:53 -05:00
|
|
|
PUT my_index/_doc/1
|
2016-05-19 13:42:23 -04:00
|
|
|
{
|
|
|
|
"title": "Quick Foxes" <1>
|
|
|
|
}
|
2013-08-28 19:24:34 -04:00
|
|
|
|
2016-05-19 13:42:23 -04:00
|
|
|
POST my_index/_refresh
|
|
|
|
|
|
|
|
GET my_index/_search
|
|
|
|
{
|
|
|
|
"query": {
|
|
|
|
"match": {
|
|
|
|
"title": {
|
|
|
|
"query": "Quick Fo", <2>
|
|
|
|
"operator": "and"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
-----------------------------------
|
2013-08-28 19:24:34 -04:00
|
|
|
|
2016-05-19 13:42:23 -04:00
|
|
|
<1> The `autocomplete` analyzer indexes the terms `[qu, qui, quic, quick, fo, fox, foxe, foxes]`.
|
|
|
|
<2> The `autocomplete_search` analyzer searches for the terms `[quick, fo]`, both of which appear in the index.
|
|
|
|
|
|
|
|
/////////////////////
|
2013-08-28 19:24:34 -04:00
|
|
|
|
2019-09-06 16:09:09 -04:00
|
|
|
[source,console-result]
|
2016-05-19 13:42:23 -04:00
|
|
|
----------------------------
|
|
|
|
{
|
|
|
|
"took": $body.took,
|
|
|
|
"timed_out": false,
|
|
|
|
"_shards": {
|
2018-05-14 12:22:35 -04:00
|
|
|
"total": 1,
|
|
|
|
"successful": 1,
|
2017-07-12 16:19:20 -04:00
|
|
|
"skipped" : 0,
|
2016-05-19 13:42:23 -04:00
|
|
|
"failed": 0
|
|
|
|
},
|
|
|
|
"hits": {
|
2018-12-05 13:49:06 -05:00
|
|
|
"total" : {
|
|
|
|
"value": 1,
|
|
|
|
"relation": "eq"
|
|
|
|
},
|
2017-06-15 03:52:07 -04:00
|
|
|
"max_score": 0.5753642,
|
2016-05-19 13:42:23 -04:00
|
|
|
"hits": [
|
|
|
|
{
|
|
|
|
"_index": "my_index",
|
2017-12-14 11:47:53 -05:00
|
|
|
"_type": "_doc",
|
2016-05-19 13:42:23 -04:00
|
|
|
"_id": "1",
|
2017-06-15 03:52:07 -04:00
|
|
|
"_score": 0.5753642,
|
2016-05-19 13:42:23 -04:00
|
|
|
"_source": {
|
|
|
|
"title": "Quick Foxes"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
----------------------------
|
|
|
|
// TESTRESPONSE[s/"took".*/"took": "$body.took",/]
|
|
|
|
/////////////////////
|