mirror of
https://github.com/go-gitea/gitea
synced 2025-01-18 10:37:42 +01:00
5dbf36f356
* Issue search support elasticsearch * Fix lint * Add indexer name on app.ini * add a warnning on SearchIssuesByKeyword * improve code
593 lines
18 KiB
Go
593 lines
18 KiB
Go
// 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 (
|
|
"fmt"
|
|
)
|
|
|
|
// SearchSource enables users to build the search source.
|
|
// It resembles the SearchSourceBuilder in Elasticsearch.
|
|
type SearchSource struct {
|
|
query Query // query
|
|
postQuery Query // post_filter
|
|
sliceQuery Query // slice
|
|
from int // from
|
|
size int // size
|
|
explain *bool // explain
|
|
version *bool // version
|
|
seqNoAndPrimaryTerm *bool // seq_no_primary_term
|
|
sorters []Sorter // sort
|
|
trackScores *bool // track_scores
|
|
trackTotalHits interface{} // track_total_hits
|
|
searchAfterSortValues []interface{} // search_after
|
|
minScore *float64 // min_score
|
|
timeout string // timeout
|
|
terminateAfter *int // terminate_after
|
|
storedFieldNames []string // stored_fields
|
|
docvalueFields DocvalueFields // docvalue_fields
|
|
scriptFields []*ScriptField // script_fields
|
|
fetchSourceContext *FetchSourceContext // _source
|
|
aggregations map[string]Aggregation // aggregations / aggs
|
|
highlight *Highlight // highlight
|
|
globalSuggestText string
|
|
suggesters []Suggester // suggest
|
|
rescores []*Rescore // rescore
|
|
defaultRescoreWindowSize *int
|
|
indexBoosts map[string]float64 // indices_boost
|
|
stats []string // stats
|
|
innerHits map[string]*InnerHit
|
|
collapse *CollapseBuilder // collapse
|
|
profile bool // profile
|
|
// TODO extBuilders []SearchExtBuilder // ext
|
|
}
|
|
|
|
// NewSearchSource initializes a new SearchSource.
|
|
func NewSearchSource() *SearchSource {
|
|
return &SearchSource{
|
|
from: -1,
|
|
size: -1,
|
|
aggregations: make(map[string]Aggregation),
|
|
indexBoosts: make(map[string]float64),
|
|
innerHits: make(map[string]*InnerHit),
|
|
}
|
|
}
|
|
|
|
// Query sets the query to use with this search source.
|
|
func (s *SearchSource) Query(query Query) *SearchSource {
|
|
s.query = query
|
|
return s
|
|
}
|
|
|
|
// Profile specifies that this search source should activate the
|
|
// Profile API for queries made on it.
|
|
func (s *SearchSource) Profile(profile bool) *SearchSource {
|
|
s.profile = profile
|
|
return s
|
|
}
|
|
|
|
// PostFilter will be executed after the query has been executed and
|
|
// only affects the search hits, not the aggregations.
|
|
// This filter is always executed as the last filtering mechanism.
|
|
func (s *SearchSource) PostFilter(postFilter Query) *SearchSource {
|
|
s.postQuery = postFilter
|
|
return s
|
|
}
|
|
|
|
// Slice allows partitioning the documents in multiple slices.
|
|
// It is e.g. used to slice a scroll operation, supported in
|
|
// Elasticsearch 5.0 or later.
|
|
// See https://www.elastic.co/guide/en/elasticsearch/reference/7.0/search-request-scroll.html#sliced-scroll
|
|
// for details.
|
|
func (s *SearchSource) Slice(sliceQuery Query) *SearchSource {
|
|
s.sliceQuery = sliceQuery
|
|
return s
|
|
}
|
|
|
|
// From index to start the search from. Defaults to 0.
|
|
func (s *SearchSource) From(from int) *SearchSource {
|
|
s.from = from
|
|
return s
|
|
}
|
|
|
|
// Size is the number of search hits to return. Defaults to 10.
|
|
func (s *SearchSource) Size(size int) *SearchSource {
|
|
s.size = size
|
|
return s
|
|
}
|
|
|
|
// MinScore sets the minimum score below which docs will be filtered out.
|
|
func (s *SearchSource) MinScore(minScore float64) *SearchSource {
|
|
s.minScore = &minScore
|
|
return s
|
|
}
|
|
|
|
// Explain indicates whether each search hit should be returned with
|
|
// an explanation of the hit (ranking).
|
|
func (s *SearchSource) Explain(explain bool) *SearchSource {
|
|
s.explain = &explain
|
|
return s
|
|
}
|
|
|
|
// Version indicates whether each search hit should be returned with
|
|
// a version associated to it.
|
|
func (s *SearchSource) Version(version bool) *SearchSource {
|
|
s.version = &version
|
|
return s
|
|
}
|
|
|
|
// SeqNoAndPrimaryTerm indicates whether SearchHits should be returned with the
|
|
// sequence number and primary term of the last modification of the document.
|
|
func (s *SearchSource) SeqNoAndPrimaryTerm(enabled bool) *SearchSource {
|
|
s.seqNoAndPrimaryTerm = &enabled
|
|
return s
|
|
}
|
|
|
|
// Timeout controls how long a search is allowed to take, e.g. "1s" or "500ms".
|
|
func (s *SearchSource) Timeout(timeout string) *SearchSource {
|
|
s.timeout = timeout
|
|
return s
|
|
}
|
|
|
|
// TimeoutInMillis controls how many milliseconds a search is allowed
|
|
// to take before it is canceled.
|
|
func (s *SearchSource) TimeoutInMillis(timeoutInMillis int) *SearchSource {
|
|
s.timeout = fmt.Sprintf("%dms", timeoutInMillis)
|
|
return s
|
|
}
|
|
|
|
// TerminateAfter specifies the maximum number of documents to collect for
|
|
// each shard, upon reaching which the query execution will terminate early.
|
|
func (s *SearchSource) TerminateAfter(terminateAfter int) *SearchSource {
|
|
s.terminateAfter = &terminateAfter
|
|
return s
|
|
}
|
|
|
|
// Sort adds a sort order.
|
|
func (s *SearchSource) Sort(field string, ascending bool) *SearchSource {
|
|
s.sorters = append(s.sorters, SortInfo{Field: field, Ascending: ascending})
|
|
return s
|
|
}
|
|
|
|
// SortWithInfo adds a sort order.
|
|
func (s *SearchSource) SortWithInfo(info SortInfo) *SearchSource {
|
|
s.sorters = append(s.sorters, info)
|
|
return s
|
|
}
|
|
|
|
// SortBy adds a sort order.
|
|
func (s *SearchSource) SortBy(sorter ...Sorter) *SearchSource {
|
|
s.sorters = append(s.sorters, sorter...)
|
|
return s
|
|
}
|
|
|
|
func (s *SearchSource) hasSort() bool {
|
|
return len(s.sorters) > 0
|
|
}
|
|
|
|
// TrackScores is applied when sorting and controls if scores will be
|
|
// tracked as well. Defaults to false.
|
|
func (s *SearchSource) TrackScores(trackScores bool) *SearchSource {
|
|
s.trackScores = &trackScores
|
|
return s
|
|
}
|
|
|
|
// TrackTotalHits controls how the total number of hits should be tracked.
|
|
// Defaults to 10000 which will count the total hit accurately up to 10,000 hits.
|
|
//
|
|
// See https://www.elastic.co/guide/en/elasticsearch/reference/7.0/search-request-track-total-hits.html
|
|
// for details.
|
|
func (s *SearchSource) TrackTotalHits(trackTotalHits interface{}) *SearchSource {
|
|
s.trackTotalHits = trackTotalHits
|
|
return s
|
|
}
|
|
|
|
// SearchAfter allows a different form of pagination by using a live cursor,
|
|
// using the results of the previous page to help the retrieval of the next.
|
|
//
|
|
// See https://www.elastic.co/guide/en/elasticsearch/reference/7.0/search-request-search-after.html
|
|
func (s *SearchSource) SearchAfter(sortValues ...interface{}) *SearchSource {
|
|
s.searchAfterSortValues = append(s.searchAfterSortValues, sortValues...)
|
|
return s
|
|
}
|
|
|
|
// Aggregation adds an aggreation to perform as part of the search.
|
|
func (s *SearchSource) Aggregation(name string, aggregation Aggregation) *SearchSource {
|
|
s.aggregations[name] = aggregation
|
|
return s
|
|
}
|
|
|
|
// DefaultRescoreWindowSize sets the rescore window size for rescores
|
|
// that don't specify their window.
|
|
func (s *SearchSource) DefaultRescoreWindowSize(defaultRescoreWindowSize int) *SearchSource {
|
|
s.defaultRescoreWindowSize = &defaultRescoreWindowSize
|
|
return s
|
|
}
|
|
|
|
// Highlight adds highlighting to the search.
|
|
func (s *SearchSource) Highlight(highlight *Highlight) *SearchSource {
|
|
s.highlight = highlight
|
|
return s
|
|
}
|
|
|
|
// Highlighter returns the highlighter.
|
|
func (s *SearchSource) Highlighter() *Highlight {
|
|
if s.highlight == nil {
|
|
s.highlight = NewHighlight()
|
|
}
|
|
return s.highlight
|
|
}
|
|
|
|
// GlobalSuggestText defines the global text to use with all suggesters.
|
|
// This avoids repetition.
|
|
func (s *SearchSource) GlobalSuggestText(text string) *SearchSource {
|
|
s.globalSuggestText = text
|
|
return s
|
|
}
|
|
|
|
// Suggester adds a suggester to the search.
|
|
func (s *SearchSource) Suggester(suggester Suggester) *SearchSource {
|
|
s.suggesters = append(s.suggesters, suggester)
|
|
return s
|
|
}
|
|
|
|
// Rescorer adds a rescorer to the search.
|
|
func (s *SearchSource) Rescorer(rescore *Rescore) *SearchSource {
|
|
s.rescores = append(s.rescores, rescore)
|
|
return s
|
|
}
|
|
|
|
// ClearRescorers removes all rescorers from the search.
|
|
func (s *SearchSource) ClearRescorers() *SearchSource {
|
|
s.rescores = make([]*Rescore, 0)
|
|
return s
|
|
}
|
|
|
|
// FetchSource indicates whether the response should contain the stored
|
|
// _source for every hit.
|
|
func (s *SearchSource) FetchSource(fetchSource bool) *SearchSource {
|
|
if s.fetchSourceContext == nil {
|
|
s.fetchSourceContext = NewFetchSourceContext(fetchSource)
|
|
} else {
|
|
s.fetchSourceContext.SetFetchSource(fetchSource)
|
|
}
|
|
return s
|
|
}
|
|
|
|
// FetchSourceContext indicates how the _source should be fetched.
|
|
func (s *SearchSource) FetchSourceContext(fetchSourceContext *FetchSourceContext) *SearchSource {
|
|
s.fetchSourceContext = fetchSourceContext
|
|
return s
|
|
}
|
|
|
|
// FetchSourceIncludeExclude specifies that _source should be returned
|
|
// with each hit, where "include" and "exclude" serve as a simple wildcard
|
|
// matcher that gets applied to its fields
|
|
// (e.g. include := []string{"obj1.*","obj2.*"}, exclude := []string{"description.*"}).
|
|
func (s *SearchSource) FetchSourceIncludeExclude(include, exclude []string) *SearchSource {
|
|
s.fetchSourceContext = NewFetchSourceContext(true).
|
|
Include(include...).
|
|
Exclude(exclude...)
|
|
return s
|
|
}
|
|
|
|
// NoStoredFields indicates that no fields should be loaded, resulting in only
|
|
// id and type to be returned per field.
|
|
func (s *SearchSource) NoStoredFields() *SearchSource {
|
|
s.storedFieldNames = []string{}
|
|
return s
|
|
}
|
|
|
|
// StoredField adds a single field to load and return (note, must be stored) as
|
|
// part of the search request. If none are specified, the source of the
|
|
// document will be returned.
|
|
func (s *SearchSource) StoredField(storedFieldName string) *SearchSource {
|
|
s.storedFieldNames = append(s.storedFieldNames, storedFieldName)
|
|
return s
|
|
}
|
|
|
|
// StoredFields sets the fields to load and return as part of the search request.
|
|
// If none are specified, the source of the document will be returned.
|
|
func (s *SearchSource) StoredFields(storedFieldNames ...string) *SearchSource {
|
|
s.storedFieldNames = append(s.storedFieldNames, storedFieldNames...)
|
|
return s
|
|
}
|
|
|
|
// DocvalueField adds a single field to load from the field data cache
|
|
// and return as part of the search request.
|
|
func (s *SearchSource) DocvalueField(fieldDataField string) *SearchSource {
|
|
s.docvalueFields = append(s.docvalueFields, DocvalueField{Field: fieldDataField})
|
|
return s
|
|
}
|
|
|
|
// DocvalueField adds a single docvalue field to load from the field data cache
|
|
// and return as part of the search request.
|
|
func (s *SearchSource) DocvalueFieldWithFormat(fieldDataFieldWithFormat DocvalueField) *SearchSource {
|
|
s.docvalueFields = append(s.docvalueFields, fieldDataFieldWithFormat)
|
|
return s
|
|
}
|
|
|
|
// DocvalueFields adds one or more fields to load from the field data cache
|
|
// and return as part of the search request.
|
|
func (s *SearchSource) DocvalueFields(docvalueFields ...string) *SearchSource {
|
|
for _, f := range docvalueFields {
|
|
s.docvalueFields = append(s.docvalueFields, DocvalueField{Field: f})
|
|
}
|
|
return s
|
|
}
|
|
|
|
// DocvalueFields adds one or more docvalue fields to load from the field data cache
|
|
// and return as part of the search request.
|
|
func (s *SearchSource) DocvalueFieldsWithFormat(docvalueFields ...DocvalueField) *SearchSource {
|
|
s.docvalueFields = append(s.docvalueFields, docvalueFields...)
|
|
return s
|
|
}
|
|
|
|
// ScriptField adds a single script field with the provided script.
|
|
func (s *SearchSource) ScriptField(scriptField *ScriptField) *SearchSource {
|
|
s.scriptFields = append(s.scriptFields, scriptField)
|
|
return s
|
|
}
|
|
|
|
// ScriptFields adds one or more script fields with the provided scripts.
|
|
func (s *SearchSource) ScriptFields(scriptFields ...*ScriptField) *SearchSource {
|
|
s.scriptFields = append(s.scriptFields, scriptFields...)
|
|
return s
|
|
}
|
|
|
|
// IndexBoost sets the boost that a specific index will receive when the
|
|
// query is executed against it.
|
|
func (s *SearchSource) IndexBoost(index string, boost float64) *SearchSource {
|
|
s.indexBoosts[index] = boost
|
|
return s
|
|
}
|
|
|
|
// Stats group this request will be aggregated under.
|
|
func (s *SearchSource) Stats(statsGroup ...string) *SearchSource {
|
|
s.stats = append(s.stats, statsGroup...)
|
|
return s
|
|
}
|
|
|
|
// InnerHit adds an inner hit to return with the result.
|
|
func (s *SearchSource) InnerHit(name string, innerHit *InnerHit) *SearchSource {
|
|
s.innerHits[name] = innerHit
|
|
return s
|
|
}
|
|
|
|
// Collapse adds field collapsing.
|
|
func (s *SearchSource) Collapse(collapse *CollapseBuilder) *SearchSource {
|
|
s.collapse = collapse
|
|
return s
|
|
}
|
|
|
|
// Source returns the serializable JSON for the source builder.
|
|
func (s *SearchSource) Source() (interface{}, error) {
|
|
source := make(map[string]interface{})
|
|
|
|
if s.from != -1 {
|
|
source["from"] = s.from
|
|
}
|
|
if s.size != -1 {
|
|
source["size"] = s.size
|
|
}
|
|
if s.timeout != "" {
|
|
source["timeout"] = s.timeout
|
|
}
|
|
if s.terminateAfter != nil {
|
|
source["terminate_after"] = *s.terminateAfter
|
|
}
|
|
if s.query != nil {
|
|
src, err := s.query.Source()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
source["query"] = src
|
|
}
|
|
if s.postQuery != nil {
|
|
src, err := s.postQuery.Source()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
source["post_filter"] = src
|
|
}
|
|
if s.minScore != nil {
|
|
source["min_score"] = *s.minScore
|
|
}
|
|
if s.version != nil {
|
|
source["version"] = *s.version
|
|
}
|
|
if s.explain != nil {
|
|
source["explain"] = *s.explain
|
|
}
|
|
if s.profile {
|
|
source["profile"] = s.profile
|
|
}
|
|
if s.fetchSourceContext != nil {
|
|
src, err := s.fetchSourceContext.Source()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
source["_source"] = src
|
|
}
|
|
if s.storedFieldNames != nil {
|
|
switch len(s.storedFieldNames) {
|
|
case 1:
|
|
source["stored_fields"] = s.storedFieldNames[0]
|
|
default:
|
|
source["stored_fields"] = s.storedFieldNames
|
|
}
|
|
}
|
|
if len(s.docvalueFields) > 0 {
|
|
src, err := s.docvalueFields.Source()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
source["docvalue_fields"] = src
|
|
}
|
|
if len(s.scriptFields) > 0 {
|
|
sfmap := make(map[string]interface{})
|
|
for _, scriptField := range s.scriptFields {
|
|
src, err := scriptField.Source()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
sfmap[scriptField.FieldName] = src
|
|
}
|
|
source["script_fields"] = sfmap
|
|
}
|
|
if len(s.sorters) > 0 {
|
|
var sortarr []interface{}
|
|
for _, sorter := range s.sorters {
|
|
src, err := sorter.Source()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
sortarr = append(sortarr, src)
|
|
}
|
|
source["sort"] = sortarr
|
|
}
|
|
if v := s.trackScores; v != nil {
|
|
source["track_scores"] = *v
|
|
}
|
|
if v := s.trackTotalHits; v != nil {
|
|
source["track_total_hits"] = v
|
|
}
|
|
if len(s.searchAfterSortValues) > 0 {
|
|
source["search_after"] = s.searchAfterSortValues
|
|
}
|
|
if s.sliceQuery != nil {
|
|
src, err := s.sliceQuery.Source()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
source["slice"] = src
|
|
}
|
|
if len(s.indexBoosts) > 0 {
|
|
source["indices_boost"] = s.indexBoosts
|
|
}
|
|
if len(s.aggregations) > 0 {
|
|
aggsMap := make(map[string]interface{})
|
|
for name, aggregate := range s.aggregations {
|
|
src, err := aggregate.Source()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
aggsMap[name] = src
|
|
}
|
|
source["aggregations"] = aggsMap
|
|
}
|
|
if s.highlight != nil {
|
|
src, err := s.highlight.Source()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
source["highlight"] = src
|
|
}
|
|
if len(s.suggesters) > 0 {
|
|
suggesters := make(map[string]interface{})
|
|
for _, s := range s.suggesters {
|
|
src, err := s.Source(false)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
suggesters[s.Name()] = src
|
|
}
|
|
if s.globalSuggestText != "" {
|
|
suggesters["text"] = s.globalSuggestText
|
|
}
|
|
source["suggest"] = suggesters
|
|
}
|
|
if len(s.rescores) > 0 {
|
|
// Strip empty rescores from request
|
|
var rescores []*Rescore
|
|
for _, r := range s.rescores {
|
|
if !r.IsEmpty() {
|
|
rescores = append(rescores, r)
|
|
}
|
|
}
|
|
if len(rescores) == 1 {
|
|
rescores[0].defaultRescoreWindowSize = s.defaultRescoreWindowSize
|
|
src, err := rescores[0].Source()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
source["rescore"] = src
|
|
} else {
|
|
var slice []interface{}
|
|
for _, r := range rescores {
|
|
r.defaultRescoreWindowSize = s.defaultRescoreWindowSize
|
|
src, err := r.Source()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
slice = append(slice, src)
|
|
}
|
|
source["rescore"] = slice
|
|
}
|
|
}
|
|
if len(s.stats) > 0 {
|
|
source["stats"] = s.stats
|
|
}
|
|
// TODO ext builders
|
|
|
|
if s.collapse != nil {
|
|
src, err := s.collapse.Source()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
source["collapse"] = src
|
|
}
|
|
|
|
if v := s.seqNoAndPrimaryTerm; v != nil {
|
|
source["seq_no_primary_term"] = *v
|
|
}
|
|
|
|
if len(s.innerHits) > 0 {
|
|
// Top-level inner hits
|
|
// See http://www.elastic.co/guide/en/elasticsearch/reference/1.5/search-request-inner-hits.html#top-level-inner-hits
|
|
// "inner_hits": {
|
|
// "<inner_hits_name>": {
|
|
// "<path|type>": {
|
|
// "<path-to-nested-object-field|child-or-parent-type>": {
|
|
// <inner_hits_body>,
|
|
// [,"inner_hits" : { [<sub_inner_hits>]+ } ]?
|
|
// }
|
|
// }
|
|
// },
|
|
// [,"<inner_hits_name_2>" : { ... } ]*
|
|
// }
|
|
m := make(map[string]interface{})
|
|
for name, hit := range s.innerHits {
|
|
if hit.path != "" {
|
|
src, err := hit.Source()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
path := make(map[string]interface{})
|
|
path[hit.path] = src
|
|
m[name] = map[string]interface{}{
|
|
"path": path,
|
|
}
|
|
} else if hit.typ != "" {
|
|
src, err := hit.Source()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
typ := make(map[string]interface{})
|
|
typ[hit.typ] = src
|
|
m[name] = map[string]interface{}{
|
|
"type": typ,
|
|
}
|
|
} else {
|
|
// TODO the Java client throws here, because either path or typ must be specified
|
|
_ = m
|
|
}
|
|
}
|
|
source["inner_hits"] = m
|
|
}
|
|
|
|
return source, nil
|
|
}
|