123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124 |
- // Copyright 2012-present Oliver Eilhard. All rights reserved.
- // Use of this source code is governed by a MIT-license.
- // See http://olivere.mit-license.org/license.txt for details.
- package elastic
- import "errors"
- // SuggesterContextQuery is used to define context information within
- // a suggestion request.
- type SuggesterContextQuery interface {
- Source() (interface{}, error)
- }
- // ContextSuggester is a fast suggester for e.g. type-ahead completion that supports filtering and boosting based on contexts.
- // See https://www.elastic.co/guide/en/elasticsearch/reference/current/suggester-context.html
- // for more details.
- type ContextSuggester struct {
- Suggester
- name string
- prefix string
- field string
- size *int
- contextQueries []SuggesterContextQuery
- }
- // Creates a new context suggester.
- func NewContextSuggester(name string) *ContextSuggester {
- return &ContextSuggester{
- name: name,
- contextQueries: make([]SuggesterContextQuery, 0),
- }
- }
- func (q *ContextSuggester) Name() string {
- return q.name
- }
- func (q *ContextSuggester) Prefix(prefix string) *ContextSuggester {
- q.prefix = prefix
- return q
- }
- func (q *ContextSuggester) Field(field string) *ContextSuggester {
- q.field = field
- return q
- }
- func (q *ContextSuggester) Size(size int) *ContextSuggester {
- q.size = &size
- return q
- }
- func (q *ContextSuggester) ContextQuery(query SuggesterContextQuery) *ContextSuggester {
- q.contextQueries = append(q.contextQueries, query)
- return q
- }
- func (q *ContextSuggester) ContextQueries(queries ...SuggesterContextQuery) *ContextSuggester {
- q.contextQueries = append(q.contextQueries, queries...)
- return q
- }
- // contextSuggesterRequest is necessary because the order in which
- // the JSON elements are routed to Elasticsearch is relevant.
- // We got into trouble when using plain maps because the text element
- // needs to go before the completion element.
- type contextSuggesterRequest struct {
- Prefix string `json:"prefix"`
- Completion interface{} `json:"completion"`
- }
- // Creates the source for the context suggester.
- func (q *ContextSuggester) Source(includeName bool) (interface{}, error) {
- cs := &contextSuggesterRequest{}
- if q.prefix != "" {
- cs.Prefix = q.prefix
- }
- suggester := make(map[string]interface{})
- cs.Completion = suggester
- if q.field != "" {
- suggester["field"] = q.field
- }
- if q.size != nil {
- suggester["size"] = *q.size
- }
- switch len(q.contextQueries) {
- case 0:
- case 1:
- src, err := q.contextQueries[0].Source()
- if err != nil {
- return nil, err
- }
- suggester["context"] = src
- default:
- ctxq := make(map[string]interface{})
- for _, query := range q.contextQueries {
- src, err := query.Source()
- if err != nil {
- return nil, err
- }
- // Merge the dictionary into ctxq
- m, ok := src.(map[string]interface{})
- if !ok {
- return nil, errors.New("elastic: context query is not a map")
- }
- for k, v := range m {
- ctxq[k] = v
- }
- }
- suggester["contexts"] = ctxq
- }
- if !includeName {
- return cs, nil
- }
- source := make(map[string]interface{})
- source[q.name] = cs
- return source, nil
- }
|