Koha::SearchEngine::Elasticsearch::QueryBuilder - constructs elasticsearch query objects from user-supplied queries
This provides the functions that take a user-supplied search query, and provides something that can be given to elasticsearch to get answers.
use Koha::SearchEngine::Elasticsearch::QueryBuilder; $builder = Koha::SearchEngine::Elasticsearch->new({ index => $index }); my $simple_query = $builder->build_query("hello"); # This is currently undocumented because the original code is undocumented my $adv_query = $builder->build_advanced_query($indexes, $operands, $operators);
my $simple_query = $builder->build_query("hello", %options)
This will build a query that can be issued to elasticsearch from the provided string input. This expects a lucene style search form (see http://www.elasticsearch.org/guide/en/elasticsearch/reference/current/query-dsl-query-string-query.html#query-string-syntax for details.)
It'll make an attempt to respect the various query options.
Additional options can be provided with the %options
hash.
This should be an arrayref of hashrefs, each containing a field
and an direction
(optional, defaults to asc
.) The results will be sorted according to these values. Valid values for direction
are 'asc' and 'desc'.
my $browse_query = $builder->build_browse_query($field, $query);
This performs a "starts with" style query on a particular field. The field to be searched must have been indexed with an appropriate mapping as a "phrase" subfield, which pretty much everything has.
my ( $error, $query, $simple_query, $query_cgi, $query_desc, $limit, $limit_cgi, $limit_desc, $stopwords_removed, $query_type ) = $builder->build_query_compat( \@operators, \@operands, \@indexes, \@limits, \@sort_by, $scan, $lang );
This handles a search using the same api as C4::Search::buildQuery does.
A very simple query will go in with $operands
set to ['query'], and $sort_by
set to ['pubdate_dsc']. This simple case will return with $query
set to something that can perform the search, $simple_query
set to just the search term, $query_cgi
set to something that can reproduce this search, and $query_desc
set to something else.
my $query = $builder->build_authorities_query(\%search);
This takes a nice description of an authority search and turns it into a black-box query that can then be passed to the appropriate searcher.
The search description is a hashref that looks something like:
{ searches => [ { where => 'Heading', # search the main entry operator => 'exact', # require an exact match value => 'frogs', # the search string }, { where => '', # search all entries operator => '', # default keyword, right truncation value => 'pond', }, ], sort => { field => 'Heading', order => 'desc', }, authtypecode => 'TOPIC_TERM', }
my ($query) = $builder->build_authorities_query_compat( \@marclist, \@and_or, \@excluding, \@operator, \@value, $authtypecode, $orderby );
This builds a query for searching for authorities, in the style of C4::AuthoritiesMarc::SearchAuthorities.
Arguments:
An arrayref containing where the particular term should be searched for. Options are: mainmainentry, mainentry, match, match-heading, see-from, and thesaurus. If left blank, any field is used.
Totally ignored. It is never used in C4::AuthoritiesMarc::SearchAuthorities.
Also ignored.
What form of search to do. Options are: is (phrase, no truncation, whole field must match), = (number exact match), exact (phrase, no truncation, whole field must match). If left blank, then word list, right truncated, anywhere is used.
The actual user-provided string value to search for.
The authority type code to search within. If blank, then all will be searched.
The order to sort the results by. Options are Relevance, HeadingAsc, HeadingDsc, AuthidAsc, AuthidDsc.
marclist, operator, and value must be the same length, and the values at index /i/ all relate to each other.
This returns a query, which is a black box object that can be passed to the appropriate search object.
my @sort_params = _convert_sort_fields(@sort_by)
Converts the zebra-style sort index information into elasticsearch-style.
@sort_by
is the same as presented to build_query_compat, and it returns something that can be sent to build_query.
my @index_params = $self->_convert_index_fields(@indexes);
Converts zebra-style search index notation into elasticsearch-style.
@indexes
is an array of index names, as presented to build_query_compat, and it returns something that can be sent to build_query.
TODO: this will pull from the elasticsearch mappings table to figure out types.
my @searches = $self->_convert_index_strings(@searches);
Similar to _convert_index_fields, this takes strings of the form field:search term and rewrites the field from zebra-style to elasticsearch-style. Anything it doesn't understand is returned verbatim.
my $search = $self->_convert_index_strings_freeform($search);
This is similar to _convert_index_strings, however it'll search out the things to change within the string. So it can handle strings such as (su:foo) AND (su:bar)
, converting the su
appropriately.
If there is something of the form "su,complete-subfield" or something, the second part is stripped off as we can't yet handle that. Making it work will have to wait for a real query parser.
my $str = $self->_modify_string_by_type(%index_field);
If you have a search term (operand) and a type (phrase, right-truncated), this will convert the string to have the function in lucene search terms, e.g. wrapping quotes around it.
my $query_str = $self->_join_queries(@query_parts);
This takes a list of query parts, that might be search terms on their own, or booleaned together, or specifying fields, or whatever, wraps them in parentheses, and ANDs them all together. Suitable for feeding to the ES query string query.
Note: doesn't AND them together if they specify an index that starts with "mc" as that was a special case in the original code for dealing with multiple choice options (you can't search for something that has an itype of A and and itype of B otherwise.)
my @phrased_queries = $self->_make_phrases(@query_parts);
This takes the supplied queries and forces them to be phrases by wrapping quotes around them. It understands field prefixes, e.g. 'subject:' and puts the quotes outside of them if they're there.
my @query_strings = $self->_create_query_string(@queries);
Given a list of hashrefs, it will turn them into a lucene-style query string. The hash should contain field, type (both for the indexes), operator, and operand.
my $term = $self->_clean_search_term($term);
This cleans a search term by removing any funny characters that may upset ES and give us an error. It also calls _convert_index_strings_freeform to ensure those parts are correct.
my $limits = $self->_fix_limit_special_cases($limits);
This converts any special cases that the limit specifications have into things that are more readily processable by the rest of the code.
The argument should be an arrayref, and it'll return an arrayref.
my $field = $self->_sort_field($field);
Given a field name, this works out what the actual name of the field to sort on should be. A '__sort' suffix is added for fields with a sort version, and for text fields either '.phrase' (for sortable versions) or '.raw' is appended to avoid sorting on a tokenized value.
my $query = $self->_truncate_terms($query);
Given a string query this function appends '*' wildcard to all terms except operands and double quoted strings.