- package kv
-
- import (
- "context"
- "errors"
- "fmt"
- "sort"
- "strconv"
- "strings"
-
- "github.com/google/orderedcode"
- dbm "github.com/tendermint/tm-db"
-
- abci "github.com/tendermint/tendermint/abci/types"
- "github.com/tendermint/tendermint/internal/pubsub/query"
- "github.com/tendermint/tendermint/internal/pubsub/query/syntax"
- "github.com/tendermint/tendermint/internal/state/indexer"
- "github.com/tendermint/tendermint/types"
- )
-
- var _ indexer.BlockIndexer = (*BlockerIndexer)(nil)
-
- // BlockerIndexer implements a block indexer, indexing FinalizeBlock
- // events with an underlying KV store. Block events are indexed by their height,
- // such that matching search criteria returns the respective block height(s).
- type BlockerIndexer struct {
- store dbm.DB
- }
-
- func New(store dbm.DB) *BlockerIndexer {
- return &BlockerIndexer{
- store: store,
- }
- }
-
- // Has returns true if the given height has been indexed. An error is returned
- // upon database query failure.
- func (idx *BlockerIndexer) Has(height int64) (bool, error) {
- key, err := heightKey(height)
- if err != nil {
- return false, fmt.Errorf("failed to create block height index key: %w", err)
- }
-
- return idx.store.Has(key)
- }
-
- // Index indexes FinalizeBlock events for a given block by its height.
- // The following is indexed:
- //
- // primary key: encode(block.height | height) => encode(height)
- // FinalizeBlock events: encode(eventType.eventAttr|eventValue|height|finalize_block) => encode(height)
- func (idx *BlockerIndexer) Index(bh types.EventDataNewBlockHeader) error {
- batch := idx.store.NewBatch()
- defer batch.Close()
-
- height := bh.Header.Height
-
- // 1. index by height
- key, err := heightKey(height)
- if err != nil {
- return fmt.Errorf("failed to create block height index key: %w", err)
- }
- if err := batch.Set(key, int64ToBytes(height)); err != nil {
- return err
- }
-
- // 2. index FinalizeBlock events
- if err := idx.indexEvents(batch, bh.ResultFinalizeBlock.Events, types.EventTypeFinalizeBlock, height); err != nil {
- return fmt.Errorf("failed to index FinalizeBlock events: %w", err)
- }
-
- return batch.WriteSync()
- }
-
- // Search performs a query for block heights that match a given FinalizeBlock
- // The given query can match against zero or more block heights. In the case
- // of height queries, i.e. block.height=H, if the height is indexed, that height
- // alone will be returned. An error and nil slice is returned. Otherwise, a
- // non-nil slice and nil error is returned.
- func (idx *BlockerIndexer) Search(ctx context.Context, q *query.Query) ([]int64, error) {
- results := make([]int64, 0)
- select {
- case <-ctx.Done():
- return results, nil
-
- default:
- }
-
- conditions := q.Syntax()
-
- // If there is an exact height query, return the result immediately
- // (if it exists).
- height, ok := lookForHeight(conditions)
- if ok {
- ok, err := idx.Has(height)
- if err != nil {
- return nil, err
- }
-
- if ok {
- return []int64{height}, nil
- }
-
- return results, nil
- }
-
- var heightsInitialized bool
- filteredHeights := make(map[string][]byte)
-
- // conditions to skip because they're handled before "everything else"
- skipIndexes := make([]int, 0)
-
- // Extract ranges. If both upper and lower bounds exist, it's better to get
- // them in order as to not iterate over kvs that are not within range.
- ranges, rangeIndexes := indexer.LookForRanges(conditions)
- if len(ranges) > 0 {
- skipIndexes = append(skipIndexes, rangeIndexes...)
-
- for _, qr := range ranges {
- prefix, err := orderedcode.Append(nil, qr.Key)
- if err != nil {
- return nil, fmt.Errorf("failed to create prefix key: %w", err)
- }
-
- if !heightsInitialized {
- filteredHeights, err = idx.matchRange(ctx, qr, prefix, filteredHeights, true)
- if err != nil {
- return nil, err
- }
-
- heightsInitialized = true
-
- // Ignore any remaining conditions if the first condition resulted in no
- // matches (assuming implicit AND operand).
- if len(filteredHeights) == 0 {
- break
- }
- } else {
- filteredHeights, err = idx.matchRange(ctx, qr, prefix, filteredHeights, false)
- if err != nil {
- return nil, err
- }
- }
- }
- }
-
- // for all other conditions
- for i, c := range conditions {
- if intInSlice(i, skipIndexes) {
- continue
- }
-
- startKey, err := orderedcode.Append(nil, c.Tag, c.Arg.Value())
- if err != nil {
- return nil, err
- }
-
- if !heightsInitialized {
- filteredHeights, err = idx.match(ctx, c, startKey, filteredHeights, true)
- if err != nil {
- return nil, err
- }
-
- heightsInitialized = true
-
- // Ignore any remaining conditions if the first condition resulted in no
- // matches (assuming implicit AND operand).
- if len(filteredHeights) == 0 {
- break
- }
- } else {
- filteredHeights, err = idx.match(ctx, c, startKey, filteredHeights, false)
- if err != nil {
- return nil, err
- }
- }
- }
-
- // fetch matching heights
- results = make([]int64, 0, len(filteredHeights))
- heights:
- for _, hBz := range filteredHeights {
- h := int64FromBytes(hBz)
-
- ok, err := idx.Has(h)
- if err != nil {
- return nil, err
- }
- if ok {
- results = append(results, h)
- }
-
- select {
- case <-ctx.Done():
- break heights
-
- default:
- }
- }
-
- sort.Slice(results, func(i, j int) bool { return results[i] < results[j] })
-
- return results, nil
- }
-
- // matchRange returns all matching block heights that match a given QueryRange
- // and start key. An already filtered result (filteredHeights) is provided such
- // that any non-intersecting matches are removed.
- //
- // NOTE: The provided filteredHeights may be empty if no previous condition has
- // matched.
- func (idx *BlockerIndexer) matchRange(
- ctx context.Context,
- qr indexer.QueryRange,
- startKey []byte,
- filteredHeights map[string][]byte,
- firstRun bool,
- ) (map[string][]byte, error) {
-
- // A previous match was attempted but resulted in no matches, so we return
- // no matches (assuming AND operand).
- if !firstRun && len(filteredHeights) == 0 {
- return filteredHeights, nil
- }
-
- tmpHeights := make(map[string][]byte)
- lowerBound := qr.LowerBoundValue()
- upperBound := qr.UpperBoundValue()
-
- it, err := dbm.IteratePrefix(idx.store, startKey)
- if err != nil {
- return nil, fmt.Errorf("failed to create prefix iterator: %w", err)
- }
- defer it.Close()
-
- iter:
- for ; it.Valid(); it.Next() {
- var (
- eventValue string
- err error
- )
-
- if qr.Key == types.BlockHeightKey {
- eventValue, err = parseValueFromPrimaryKey(it.Key())
- } else {
- eventValue, err = parseValueFromEventKey(it.Key())
- }
-
- if err != nil {
- continue
- }
-
- if _, ok := qr.AnyBound().(int64); ok {
- v, err := strconv.ParseInt(eventValue, 10, 64)
- if err != nil {
- continue iter
- }
-
- include := true
- if lowerBound != nil && v < lowerBound.(int64) {
- include = false
- }
-
- if upperBound != nil && v > upperBound.(int64) {
- include = false
- }
-
- if include {
- tmpHeights[string(it.Value())] = it.Value()
- }
- }
-
- select {
- case <-ctx.Done():
- break iter
-
- default:
- }
- }
-
- if err := it.Error(); err != nil {
- return nil, err
- }
-
- if len(tmpHeights) == 0 || firstRun {
- // Either:
- //
- // 1. Regardless if a previous match was attempted, which may have had
- // results, but no match was found for the current condition, then we
- // return no matches (assuming AND operand).
- //
- // 2. A previous match was not attempted, so we return all results.
- return tmpHeights, nil
- }
-
- // Remove/reduce matches in filteredHashes that were not found in this
- // match (tmpHashes).
- for k := range filteredHeights {
- if tmpHeights[k] == nil {
- delete(filteredHeights, k)
-
- select {
- case <-ctx.Done():
- break
-
- default:
- }
- }
- }
-
- return filteredHeights, nil
- }
-
- // match returns all matching heights that meet a given query condition and start
- // key. An already filtered result (filteredHeights) is provided such that any
- // non-intersecting matches are removed.
- //
- // NOTE: The provided filteredHeights may be empty if no previous condition has
- // matched.
- func (idx *BlockerIndexer) match(
- ctx context.Context,
- c syntax.Condition,
- startKeyBz []byte,
- filteredHeights map[string][]byte,
- firstRun bool,
- ) (map[string][]byte, error) {
-
- // A previous match was attempted but resulted in no matches, so we return
- // no matches (assuming AND operand).
- if !firstRun && len(filteredHeights) == 0 {
- return filteredHeights, nil
- }
-
- tmpHeights := make(map[string][]byte)
-
- switch {
- case c.Op == syntax.TEq:
- it, err := dbm.IteratePrefix(idx.store, startKeyBz)
- if err != nil {
- return nil, fmt.Errorf("failed to create prefix iterator: %w", err)
- }
- defer it.Close()
-
- for ; it.Valid(); it.Next() {
- tmpHeights[string(it.Value())] = it.Value()
-
- if err := ctx.Err(); err != nil {
- break
- }
- }
-
- if err := it.Error(); err != nil {
- return nil, err
- }
-
- case c.Op == syntax.TExists:
- prefix, err := orderedcode.Append(nil, c.Tag)
- if err != nil {
- return nil, err
- }
-
- it, err := dbm.IteratePrefix(idx.store, prefix)
- if err != nil {
- return nil, fmt.Errorf("failed to create prefix iterator: %w", err)
- }
- defer it.Close()
-
- iterExists:
- for ; it.Valid(); it.Next() {
- tmpHeights[string(it.Value())] = it.Value()
-
- select {
- case <-ctx.Done():
- break iterExists
-
- default:
- }
- }
-
- if err := it.Error(); err != nil {
- return nil, err
- }
-
- case c.Op == syntax.TContains:
- prefix, err := orderedcode.Append(nil, c.Tag)
- if err != nil {
- return nil, err
- }
-
- it, err := dbm.IteratePrefix(idx.store, prefix)
- if err != nil {
- return nil, fmt.Errorf("failed to create prefix iterator: %w", err)
- }
- defer it.Close()
-
- iterContains:
- for ; it.Valid(); it.Next() {
- eventValue, err := parseValueFromEventKey(it.Key())
- if err != nil {
- continue
- }
-
- if strings.Contains(eventValue, c.Arg.Value()) {
- tmpHeights[string(it.Value())] = it.Value()
- }
-
- select {
- case <-ctx.Done():
- break iterContains
-
- default:
- }
- }
- if err := it.Error(); err != nil {
- return nil, err
- }
-
- default:
- return nil, errors.New("other operators should be handled already")
- }
-
- if len(tmpHeights) == 0 || firstRun {
- // Either:
- //
- // 1. Regardless if a previous match was attempted, which may have had
- // results, but no match was found for the current condition, then we
- // return no matches (assuming AND operand).
- //
- // 2. A previous match was not attempted, so we return all results.
- return tmpHeights, nil
- }
-
- // Remove/reduce matches in filteredHeights that were not found in this
- // match (tmpHeights).
- for k := range filteredHeights {
- if tmpHeights[k] == nil {
- delete(filteredHeights, k)
-
- select {
- case <-ctx.Done():
- break
-
- default:
- }
- }
- }
-
- return filteredHeights, nil
- }
-
- func (idx *BlockerIndexer) indexEvents(batch dbm.Batch, events []abci.Event, typ string, height int64) error {
- heightBz := int64ToBytes(height)
-
- for _, event := range events {
- // only index events with a non-empty type
- if len(event.Type) == 0 {
- continue
- }
-
- for _, attr := range event.Attributes {
- if len(attr.Key) == 0 {
- continue
- }
-
- // index iff the event specified index:true and it's not a reserved event
- compositeKey := fmt.Sprintf("%s.%s", event.Type, attr.Key)
- if compositeKey == types.BlockHeightKey {
- return fmt.Errorf("event type and attribute key \"%s\" is reserved; please use a different key", compositeKey)
- }
-
- if attr.GetIndex() {
- key, err := eventKey(compositeKey, typ, attr.Value, height)
- if err != nil {
- return fmt.Errorf("failed to create block index key: %w", err)
- }
-
- if err := batch.Set(key, heightBz); err != nil {
- return err
- }
- }
- }
- }
-
- return nil
- }
|