2022-09-07 16:15:54 +00:00
|
|
|
// Copyright 2022 The Matrix.org Foundation C.I.C.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
//go:build !wasm
|
|
|
|
// +build !wasm
|
|
|
|
|
|
|
|
package fulltext
|
|
|
|
|
|
|
|
import (
|
2023-03-27 09:26:52 +00:00
|
|
|
"regexp"
|
2022-09-07 16:15:54 +00:00
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/blevesearch/bleve/v2"
|
2023-03-22 13:12:06 +00:00
|
|
|
"github.com/matrix-org/dendrite/setup/process"
|
2023-04-19 14:50:33 +00:00
|
|
|
"github.com/matrix-org/gomatrixserverlib/spec"
|
2023-03-22 13:12:06 +00:00
|
|
|
|
2022-10-05 09:14:33 +00:00
|
|
|
// side effect imports to allow all possible languages
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/ar"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/cjk"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/ckb"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/da"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/de"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/en"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/es"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/fa"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/fi"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/fr"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/hi"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/hr"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/hu"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/it"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/nl"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/no"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/pt"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/ro"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/ru"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/sv"
|
|
|
|
_ "github.com/blevesearch/bleve/v2/analysis/lang/tr"
|
2022-09-07 16:15:54 +00:00
|
|
|
"github.com/blevesearch/bleve/v2/mapping"
|
2022-09-27 16:06:49 +00:00
|
|
|
|
|
|
|
"github.com/matrix-org/dendrite/setup/config"
|
2022-09-07 16:15:54 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Search contains all existing bleve.Index
|
|
|
|
type Search struct {
|
|
|
|
FulltextIndex bleve.Index
|
|
|
|
}
|
|
|
|
|
2023-03-17 11:09:45 +00:00
|
|
|
type Indexer interface {
|
|
|
|
Index(elements ...IndexElement) error
|
|
|
|
Delete(eventID string) error
|
|
|
|
Search(term string, roomIDs, keys []string, limit, from int, orderByStreamPos bool) (*bleve.SearchResult, error)
|
2023-03-27 09:26:52 +00:00
|
|
|
GetHighlights(result *bleve.SearchResult) []string
|
2023-03-17 11:09:45 +00:00
|
|
|
Close() error
|
|
|
|
}
|
|
|
|
|
2022-09-07 16:15:54 +00:00
|
|
|
// IndexElement describes the layout of an element to index
|
|
|
|
type IndexElement struct {
|
|
|
|
EventID string
|
|
|
|
RoomID string
|
|
|
|
Content string
|
|
|
|
ContentType string
|
|
|
|
StreamPosition int64
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetContentType sets i.ContentType given an identifier
|
|
|
|
func (i *IndexElement) SetContentType(v string) {
|
|
|
|
switch v {
|
|
|
|
case "m.room.message":
|
|
|
|
i.ContentType = "content.body"
|
2023-04-19 14:50:33 +00:00
|
|
|
case spec.MRoomName:
|
2022-09-07 16:15:54 +00:00
|
|
|
i.ContentType = "content.name"
|
2023-04-19 14:50:33 +00:00
|
|
|
case spec.MRoomTopic:
|
2022-09-07 16:15:54 +00:00
|
|
|
i.ContentType = "content.topic"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// New opens a new/existing fulltext index
|
2023-03-22 13:12:06 +00:00
|
|
|
func New(processCtx *process.ProcessContext, cfg config.Fulltext) (fts *Search, err error) {
|
2022-09-07 16:15:54 +00:00
|
|
|
fts = &Search{}
|
|
|
|
fts.FulltextIndex, err = openIndex(cfg)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-03-17 11:09:45 +00:00
|
|
|
go func() {
|
2023-03-22 13:12:06 +00:00
|
|
|
processCtx.ComponentStarted()
|
|
|
|
// Wait for the processContext to be done, indicating that Dendrite is shutting down.
|
|
|
|
<-processCtx.WaitForShutdown()
|
2023-03-17 11:09:45 +00:00
|
|
|
_ = fts.Close()
|
2023-03-22 13:12:06 +00:00
|
|
|
processCtx.ComponentFinished()
|
2023-03-17 11:09:45 +00:00
|
|
|
}()
|
2022-09-07 16:15:54 +00:00
|
|
|
return fts, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close closes the fulltext index
|
|
|
|
func (f *Search) Close() error {
|
|
|
|
return f.FulltextIndex.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Index indexes the given elements
|
|
|
|
func (f *Search) Index(elements ...IndexElement) error {
|
|
|
|
batch := f.FulltextIndex.NewBatch()
|
|
|
|
|
|
|
|
for _, element := range elements {
|
|
|
|
err := batch.Index(element.EventID, element)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return f.FulltextIndex.Batch(batch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete deletes an indexed element by the eventID
|
|
|
|
func (f *Search) Delete(eventID string) error {
|
|
|
|
return f.FulltextIndex.Delete(eventID)
|
|
|
|
}
|
|
|
|
|
2023-03-27 09:26:52 +00:00
|
|
|
var highlightMatcher = regexp.MustCompile("<mark>(.*?)</mark>")
|
|
|
|
|
|
|
|
// GetHighlights extracts the highlights from a SearchResult.
|
|
|
|
func (f *Search) GetHighlights(result *bleve.SearchResult) []string {
|
|
|
|
if result == nil {
|
|
|
|
return []string{}
|
|
|
|
}
|
|
|
|
|
|
|
|
seenMatches := make(map[string]struct{})
|
|
|
|
|
|
|
|
for _, hit := range result.Hits {
|
|
|
|
if hit.Fragments == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
fragments, ok := hit.Fragments["Content"]
|
|
|
|
if !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
for _, x := range fragments {
|
|
|
|
substringMatches := highlightMatcher.FindAllStringSubmatch(x, -1)
|
|
|
|
for _, matches := range substringMatches {
|
|
|
|
for i := range matches {
|
|
|
|
if i == 0 { // skip first match, this is the complete substring match
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if _, ok := seenMatches[matches[i]]; ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
seenMatches[matches[i]] = struct{}{}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
res := make([]string, 0, len(seenMatches))
|
|
|
|
for m := range seenMatches {
|
|
|
|
res = append(res, m)
|
|
|
|
}
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
2022-09-07 16:15:54 +00:00
|
|
|
// Search searches the index given a search term, roomIDs and keys.
|
|
|
|
func (f *Search) Search(term string, roomIDs, keys []string, limit, from int, orderByStreamPos bool) (*bleve.SearchResult, error) {
|
|
|
|
qry := bleve.NewConjunctionQuery()
|
|
|
|
termQuery := bleve.NewBooleanQuery()
|
|
|
|
|
|
|
|
terms := strings.Split(term, " ")
|
|
|
|
for _, term := range terms {
|
|
|
|
matchQuery := bleve.NewMatchQuery(term)
|
|
|
|
matchQuery.SetField("Content")
|
|
|
|
termQuery.AddMust(matchQuery)
|
|
|
|
}
|
|
|
|
qry.AddQuery(termQuery)
|
|
|
|
|
|
|
|
roomQuery := bleve.NewBooleanQuery()
|
|
|
|
for _, roomID := range roomIDs {
|
|
|
|
roomSearch := bleve.NewMatchQuery(roomID)
|
|
|
|
roomSearch.SetField("RoomID")
|
|
|
|
roomQuery.AddShould(roomSearch)
|
|
|
|
}
|
|
|
|
if len(roomIDs) > 0 {
|
|
|
|
qry.AddQuery(roomQuery)
|
|
|
|
}
|
|
|
|
keyQuery := bleve.NewBooleanQuery()
|
|
|
|
for _, key := range keys {
|
|
|
|
keySearch := bleve.NewMatchQuery(key)
|
|
|
|
keySearch.SetField("ContentType")
|
|
|
|
keyQuery.AddShould(keySearch)
|
|
|
|
}
|
|
|
|
if len(keys) > 0 {
|
|
|
|
qry.AddQuery(keyQuery)
|
|
|
|
}
|
|
|
|
|
|
|
|
s := bleve.NewSearchRequestOptions(qry, limit, from, false)
|
|
|
|
s.Fields = []string{"*"}
|
|
|
|
s.SortBy([]string{"_score"})
|
|
|
|
if orderByStreamPos {
|
|
|
|
s.SortBy([]string{"-StreamPosition"})
|
|
|
|
}
|
|
|
|
|
2023-03-27 09:26:52 +00:00
|
|
|
// Highlight some words
|
|
|
|
s.Highlight = bleve.NewHighlight()
|
|
|
|
s.Highlight.Fields = []string{"Content"}
|
|
|
|
|
2022-09-07 16:15:54 +00:00
|
|
|
return f.FulltextIndex.Search(s)
|
|
|
|
}
|
|
|
|
|
|
|
|
func openIndex(cfg config.Fulltext) (bleve.Index, error) {
|
|
|
|
m := getMapping(cfg)
|
|
|
|
if cfg.InMemory {
|
|
|
|
return bleve.NewMemOnly(m)
|
|
|
|
}
|
|
|
|
if index, err := bleve.Open(string(cfg.IndexPath)); err == nil {
|
|
|
|
return index, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
index, err := bleve.New(string(cfg.IndexPath), m)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return index, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getMapping(cfg config.Fulltext) *mapping.IndexMappingImpl {
|
|
|
|
enFieldMapping := bleve.NewTextFieldMapping()
|
|
|
|
enFieldMapping.Analyzer = cfg.Language
|
|
|
|
|
|
|
|
eventMapping := bleve.NewDocumentMapping()
|
|
|
|
eventMapping.AddFieldMappingsAt("Content", enFieldMapping)
|
|
|
|
eventMapping.AddFieldMappingsAt("StreamPosition", bleve.NewNumericFieldMapping())
|
|
|
|
|
|
|
|
// Index entries as is
|
|
|
|
idFieldMapping := bleve.NewKeywordFieldMapping()
|
|
|
|
eventMapping.AddFieldMappingsAt("ContentType", idFieldMapping)
|
|
|
|
eventMapping.AddFieldMappingsAt("RoomID", idFieldMapping)
|
|
|
|
eventMapping.AddFieldMappingsAt("EventID", idFieldMapping)
|
|
|
|
|
|
|
|
indexMapping := bleve.NewIndexMapping()
|
|
|
|
indexMapping.AddDocumentMapping("Event", eventMapping)
|
|
|
|
indexMapping.DefaultType = "Event"
|
|
|
|
return indexMapping
|
|
|
|
}
|