mirror of
				https://github.com/go-gitea/gitea
				synced 2025-11-03 21:08:25 +00:00 
			
		
		
		
	* update bleve to master b17287a86f6cac923a5d886e10618df994eeb54b6724eac2e3b8dde89cfbe3a2 * remove unused pkg from dep file * change bleve from master to recent revision
		
			
				
	
	
		
			227 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			227 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
//  Copyright (c) 2014 Couchbase, Inc.
 | 
						|
//
 | 
						|
// 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.
 | 
						|
 | 
						|
package upsidedown
 | 
						|
 | 
						|
import (
 | 
						|
	"reflect"
 | 
						|
 | 
						|
	"github.com/blevesearch/bleve/document"
 | 
						|
	"github.com/blevesearch/bleve/index"
 | 
						|
	"github.com/blevesearch/bleve/index/store"
 | 
						|
)
 | 
						|
 | 
						|
var reflectStaticSizeIndexReader int
 | 
						|
 | 
						|
func init() {
 | 
						|
	var ir IndexReader
 | 
						|
	reflectStaticSizeIndexReader = int(reflect.TypeOf(ir).Size())
 | 
						|
}
 | 
						|
 | 
						|
type IndexReader struct {
 | 
						|
	index    *UpsideDownCouch
 | 
						|
	kvreader store.KVReader
 | 
						|
	docCount uint64
 | 
						|
}
 | 
						|
 | 
						|
func (i *IndexReader) TermFieldReader(term []byte, fieldName string, includeFreq, includeNorm, includeTermVectors bool) (index.TermFieldReader, error) {
 | 
						|
	fieldIndex, fieldExists := i.index.fieldCache.FieldNamed(fieldName, false)
 | 
						|
	if fieldExists {
 | 
						|
		return newUpsideDownCouchTermFieldReader(i, term, uint16(fieldIndex), includeFreq, includeNorm, includeTermVectors)
 | 
						|
	}
 | 
						|
	return newUpsideDownCouchTermFieldReader(i, []byte{ByteSeparator}, ^uint16(0), includeFreq, includeNorm, includeTermVectors)
 | 
						|
}
 | 
						|
 | 
						|
func (i *IndexReader) FieldDict(fieldName string) (index.FieldDict, error) {
 | 
						|
	return i.FieldDictRange(fieldName, nil, nil)
 | 
						|
}
 | 
						|
 | 
						|
func (i *IndexReader) FieldDictRange(fieldName string, startTerm []byte, endTerm []byte) (index.FieldDict, error) {
 | 
						|
	fieldIndex, fieldExists := i.index.fieldCache.FieldNamed(fieldName, false)
 | 
						|
	if fieldExists {
 | 
						|
		return newUpsideDownCouchFieldDict(i, uint16(fieldIndex), startTerm, endTerm)
 | 
						|
	}
 | 
						|
	return newUpsideDownCouchFieldDict(i, ^uint16(0), []byte{ByteSeparator}, []byte{})
 | 
						|
}
 | 
						|
 | 
						|
func (i *IndexReader) FieldDictPrefix(fieldName string, termPrefix []byte) (index.FieldDict, error) {
 | 
						|
	return i.FieldDictRange(fieldName, termPrefix, termPrefix)
 | 
						|
}
 | 
						|
 | 
						|
func (i *IndexReader) DocIDReaderAll() (index.DocIDReader, error) {
 | 
						|
	return newUpsideDownCouchDocIDReader(i)
 | 
						|
}
 | 
						|
 | 
						|
func (i *IndexReader) DocIDReaderOnly(ids []string) (index.DocIDReader, error) {
 | 
						|
	return newUpsideDownCouchDocIDReaderOnly(i, ids)
 | 
						|
}
 | 
						|
 | 
						|
func (i *IndexReader) Document(id string) (doc *document.Document, err error) {
 | 
						|
	// first hit the back index to confirm doc exists
 | 
						|
	var backIndexRow *BackIndexRow
 | 
						|
	backIndexRow, err = backIndexRowForDoc(i.kvreader, []byte(id))
 | 
						|
	if err != nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
	if backIndexRow == nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
	doc = document.NewDocument(id)
 | 
						|
	storedRow := NewStoredRow([]byte(id), 0, []uint64{}, 'x', nil)
 | 
						|
	storedRowScanPrefix := storedRow.ScanPrefixForDoc()
 | 
						|
	it := i.kvreader.PrefixIterator(storedRowScanPrefix)
 | 
						|
	defer func() {
 | 
						|
		if cerr := it.Close(); err == nil && cerr != nil {
 | 
						|
			err = cerr
 | 
						|
		}
 | 
						|
	}()
 | 
						|
	key, val, valid := it.Current()
 | 
						|
	for valid {
 | 
						|
		safeVal := make([]byte, len(val))
 | 
						|
		copy(safeVal, val)
 | 
						|
		var row *StoredRow
 | 
						|
		row, err = NewStoredRowKV(key, safeVal)
 | 
						|
		if err != nil {
 | 
						|
			doc = nil
 | 
						|
			return
 | 
						|
		}
 | 
						|
		if row != nil {
 | 
						|
			fieldName := i.index.fieldCache.FieldIndexed(row.field)
 | 
						|
			field := decodeFieldType(row.typ, fieldName, row.arrayPositions, row.value)
 | 
						|
			if field != nil {
 | 
						|
				doc.AddField(field)
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		it.Next()
 | 
						|
		key, val, valid = it.Current()
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (i *IndexReader) DocumentVisitFieldTerms(id index.IndexInternalID, fields []string, visitor index.DocumentFieldTermVisitor) error {
 | 
						|
	fieldsMap := make(map[uint16]string, len(fields))
 | 
						|
	for _, f := range fields {
 | 
						|
		id, ok := i.index.fieldCache.FieldNamed(f, false)
 | 
						|
		if ok {
 | 
						|
			fieldsMap[id] = f
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	tempRow := BackIndexRow{
 | 
						|
		doc: id,
 | 
						|
	}
 | 
						|
 | 
						|
	keyBuf := GetRowBuffer()
 | 
						|
	if tempRow.KeySize() > len(keyBuf) {
 | 
						|
		keyBuf = make([]byte, 2*tempRow.KeySize())
 | 
						|
	}
 | 
						|
	defer PutRowBuffer(keyBuf)
 | 
						|
	keySize, err := tempRow.KeyTo(keyBuf)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	value, err := i.kvreader.Get(keyBuf[:keySize])
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if value == nil {
 | 
						|
		return nil
 | 
						|
	}
 | 
						|
 | 
						|
	return visitBackIndexRow(value, func(field uint32, term []byte) {
 | 
						|
		if field, ok := fieldsMap[uint16(field)]; ok {
 | 
						|
			visitor(field, term)
 | 
						|
		}
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
func (i *IndexReader) Fields() (fields []string, err error) {
 | 
						|
	fields = make([]string, 0)
 | 
						|
	it := i.kvreader.PrefixIterator([]byte{'f'})
 | 
						|
	defer func() {
 | 
						|
		if cerr := it.Close(); err == nil && cerr != nil {
 | 
						|
			err = cerr
 | 
						|
		}
 | 
						|
	}()
 | 
						|
	key, val, valid := it.Current()
 | 
						|
	for valid {
 | 
						|
		var row UpsideDownCouchRow
 | 
						|
		row, err = ParseFromKeyValue(key, val)
 | 
						|
		if err != nil {
 | 
						|
			fields = nil
 | 
						|
			return
 | 
						|
		}
 | 
						|
		if row != nil {
 | 
						|
			fieldRow, ok := row.(*FieldRow)
 | 
						|
			if ok {
 | 
						|
				fields = append(fields, fieldRow.name)
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		it.Next()
 | 
						|
		key, val, valid = it.Current()
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (i *IndexReader) GetInternal(key []byte) ([]byte, error) {
 | 
						|
	internalRow := NewInternalRow(key, nil)
 | 
						|
	return i.kvreader.Get(internalRow.Key())
 | 
						|
}
 | 
						|
 | 
						|
func (i *IndexReader) DocCount() (uint64, error) {
 | 
						|
	return i.docCount, nil
 | 
						|
}
 | 
						|
 | 
						|
func (i *IndexReader) Close() error {
 | 
						|
	return i.kvreader.Close()
 | 
						|
}
 | 
						|
 | 
						|
func (i *IndexReader) ExternalID(id index.IndexInternalID) (string, error) {
 | 
						|
	return string(id), nil
 | 
						|
}
 | 
						|
 | 
						|
func (i *IndexReader) InternalID(id string) (index.IndexInternalID, error) {
 | 
						|
	return index.IndexInternalID(id), nil
 | 
						|
}
 | 
						|
 | 
						|
func incrementBytes(in []byte) []byte {
 | 
						|
	rv := make([]byte, len(in))
 | 
						|
	copy(rv, in)
 | 
						|
	for i := len(rv) - 1; i >= 0; i-- {
 | 
						|
		rv[i] = rv[i] + 1
 | 
						|
		if rv[i] != 0 {
 | 
						|
			// didn't overflow, so stop
 | 
						|
			break
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return rv
 | 
						|
}
 | 
						|
 | 
						|
func (i *IndexReader) DocValueReader(fields []string) (index.DocValueReader, error) {
 | 
						|
	return &DocValueReader{i: i, fields: fields}, nil
 | 
						|
}
 | 
						|
 | 
						|
type DocValueReader struct {
 | 
						|
	i      *IndexReader
 | 
						|
	fields []string
 | 
						|
}
 | 
						|
 | 
						|
func (dvr *DocValueReader) VisitDocValues(id index.IndexInternalID,
 | 
						|
	visitor index.DocumentFieldTermVisitor) error {
 | 
						|
	return dvr.i.DocumentVisitFieldTerms(id, dvr.fields, visitor)
 | 
						|
}
 |