Searching is naturally the most involved operation. There are many ways to do searching in elastic search and that is reflected in the higher complexity of the search DSL.
To do a simple string query search, where the search query is parsed from a single string
search in "places"->"cities" query "London"
We can search for everything by not specifying a query at all.
search in "places"->"cities"
We might want to limit the number of results and / or set the offset.
search in "places"->"cities" query "paris" start 5 limit 10
One of the great features of Elasticsearch is the number of queries it provides. Here we can use the term query to limit the search to just the state of Georgia rather than the country of Georgia.
search in "places"->"cities" query { term("state", "georgia") }
We wouldn't be able to do very much if we couldn't combine queries. So here we combine three queries, 2 "musts" that must match the documents and 1 "not" that must not match the documents. This is what ElasticSearch calls a boolean query. You'll see in this example that I don't like to vacation anywhere that is too hot, and I want to only vacation somewhere that is awesome and that where the name ends with 'cester' like Gloucester or Leicester.
search in "places"->"cities" query {
bool {
must(
regex("name", ".*cester"),
term("status", "Awesome")
) not (
term("weather", "hot")
)
}
}
It is also possible to use raw json queries. This provides more flexibility (i.e when the DSL is not complete) and enables storing queries in a separate environment (DB, cache, etc.).
search in "*" types("users", "tweets") limit 5 rawQuery {
"""{ "prefix": { "bands": { "prefix": "coldplay", "boost": 5.0, "rewrite": "yes" } } }"""
} searchType SearchType.Scan
We might want to return facets from our search. Naturally in London we'd want to search for historic landmarks and the age of those attractions and so we'd offer these as selectable facets to our lovely users.
search in "places"->"cities" query "london" facets (
facet terms "landmark" field "type",
facet range "age" field "year" range (1000->1200) to(1200) from(1400)
)
Other facet types include geo distance, query, filter, range, date, histogram. The full documentation is here.
Elasticsearch provides sorting of course. So does elastic4s. You can even include multiple sorts - rather like multiple order clauses in an SQL query.
search in "places"->"cities" query "europe" sort (
by field "name",
by field "status"
)
Aggregations are the new facets. The basic way of doing aggregations in elastic4s is this:
client.execute {
search in "index" / "type" query <somequery> aggregations(
aggregation terms "agg1" field "field1" size 20,
aggregation avg "agg2" field "field2"
}
That would create two aggregations, one for the number of terms up to the 20 most common values in "field1", and one for the avg of the values in "field2"
We can control which parts of the source are returned to us using source filtering. Let's carry on our places/cities
example, but now lets suppose the document has many more fields, such as population, foundation date,
gps coordinates. We can specify which ones are included / excludes by using the sourceInclude
and sourceExclude
methods. This is useful functionality to trim down large documents from being sent over the wire.
client.execute {
search in "places/cities" query "europe" sourceInclude("gps", "populat*") sourceExclude("denonymn", "capit*")
}
We can specify multiple includes/excludes and they recognize regular expressions. Read more in the elasticsearch docs
Other options provided are highlighting, suggestions, filters, scrolling, index boosts and scripting. See the query dsl for more information.
Since version 1.5.0, Elasticsearch has supported inner hits.
To do top level inner hits in elastic4s we can do:
search in "index" / "type" inner (
inner hit "name" path "path",
inner hit "name" `type` "type"
)
And to use inner hits on nested queries we can do:
search in "index" / "type" query {
nestedQuery("somepath") inner {
inner hits "name" from 2 size 10
}
}
We can use script fields to evaluate a field for each hit. Script fields can even operate on fields that are not stored. Script fields can include parameters which can be accessed when the script is evaluted.
We can specify the script fields in a search query through the use of the scriptfields
method:
search in "tubestops" query "wimbledon" scriptfields (
script field "fare_cost" script "doc['zone'] * fare_per_zone" params Map("fare_per_zone" -> 3.00)
)
The general form of scriptfield DSL expressions is:
script field <field_name> script <script> (lang <lang_name>){0,1} (params <Map[String,Any]>){0,1}
See the script fields section of the elasticsearch guide for greater detail of the use of script fields in searches.