pkg/index/testcases/duration.go (278 lines of code) (raw):
// Licensed to Apache Software Foundation (ASF) under one or more contributor
// license agreements. See the NOTICE file distributed with
// this work for additional information regarding copyright
// ownership. Apache Software Foundation (ASF) licenses this file to you 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 testcases implements common helpers for testing inverted and lsm indices.
package testcases
import (
"context"
"fmt"
"math"
"sort"
"testing"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
modelv1 "github.com/apache/skywalking-banyandb/api/proto/banyandb/model/v1"
"github.com/apache/skywalking-banyandb/pkg/index"
"github.com/apache/skywalking-banyandb/pkg/index/posting"
"github.com/apache/skywalking-banyandb/pkg/index/posting/roaring"
)
var duration = index.FieldKey{
SeriesID: 998,
// duration
IndexRuleID: 3,
}
// SimpleStore is subset of index.Store for testing.
type SimpleStore interface {
index.FieldIterable
index.Writer
MatchTerms(field index.Field) (list posting.List, timestamps posting.List, err error)
}
type args struct {
termRange index.RangeOpts
fieldKey index.FieldKey
orderType modelv1.Sort
}
type result struct {
items []uint64
}
// RunDuration executes duration related cases.
func RunDuration(t *testing.T, data map[int]posting.List, store SimpleStore) {
tests := []struct {
name string
want []int
args args
}{
{
name: "sort in asc order",
args: args{
fieldKey: duration,
orderType: modelv1.Sort_SORT_ASC,
},
want: []int{50, 200, 500, 1000, 2000},
},
{
name: "sort in desc order",
args: args{
fieldKey: duration,
orderType: modelv1.Sort_SORT_DESC,
},
want: []int{2000, 1000, 500, 200, 50},
},
{
name: "scan in (lower, upper) and sort in asc order",
args: args{
fieldKey: duration,
orderType: modelv1.Sort_SORT_ASC,
termRange: index.NewIntRangeOpts(50, 2000, false, false),
},
want: []int{200, 500, 1000},
},
{
name: "scan in (lower, upper) and sort in desc order",
args: args{
fieldKey: duration,
orderType: modelv1.Sort_SORT_DESC,
termRange: index.NewIntRangeOpts(50, 2000, false, false),
},
want: []int{1000, 500, 200},
},
{
name: "scan in [lower, upper] and sort in asc order",
args: args{
fieldKey: duration,
orderType: modelv1.Sort_SORT_ASC,
termRange: index.NewIntRangeOpts(200, 1000, true, true),
},
want: []int{200, 500, 1000},
},
{
name: "scan in [lower, upper] and sort in desc order",
args: args{
fieldKey: duration,
orderType: modelv1.Sort_SORT_DESC,
termRange: index.NewIntRangeOpts(200, 1000, true, true),
},
want: []int{1000, 500, 200},
},
{
name: "scan in [lower, undefined) and sort in asc order",
args: args{
fieldKey: duration,
orderType: modelv1.Sort_SORT_ASC,
termRange: index.NewIntRangeOpts(200, math.MaxInt64, true, false),
},
want: []int{200, 500, 1000, 2000},
},
{
name: "scan in [lower, undefined) and sort in desc order",
args: args{
fieldKey: duration,
orderType: modelv1.Sort_SORT_DESC,
termRange: index.NewIntRangeOpts(200, math.MaxInt64, true, false),
},
want: []int{2000, 1000, 500, 200},
},
{
name: "scan in (undefined, upper] and sort in asc order",
args: args{
fieldKey: duration,
orderType: modelv1.Sort_SORT_ASC,
termRange: index.NewIntRangeOpts(math.MinInt64, 1000, false, true),
},
want: []int{50, 200, 500, 1000},
},
{
name: "scan in (undefined, upper] and sort in desc order",
args: args{
fieldKey: duration,
orderType: modelv1.Sort_SORT_DESC,
termRange: index.NewIntRangeOpts(math.MinInt64, 1000, false, true),
},
want: []int{1000, 500, 200, 50},
},
{
name: "scan splice in (lower, upper) and sort in asc order",
args: args{
fieldKey: duration,
orderType: modelv1.Sort_SORT_ASC,
termRange: index.NewIntRangeOpts(150, 1900, false, false),
},
want: []int{200, 500, 1000},
},
{
name: "scan splice in (lower, upper) and sort in desc order",
args: args{
fieldKey: duration,
orderType: modelv1.Sort_SORT_DESC,
termRange: index.NewIntRangeOpts(150, 1900, false, false),
},
want: []int{1000, 500, 200},
},
{
name: "scan splice in [lower, upper] and sort in asc order",
args: args{
fieldKey: duration,
orderType: modelv1.Sort_SORT_ASC,
termRange: index.NewIntRangeOpts(150, 1900, true, true),
},
want: []int{200, 500, 1000},
},
{
name: "scan splice in [lower, upper] and sort in desc order",
args: args{
fieldKey: duration,
orderType: modelv1.Sort_SORT_DESC,
termRange: index.NewIntRangeOpts(150, 1900, true, true),
},
want: []int{1000, 500, 200},
},
{
name: "no field key",
args: args{},
},
{
name: "unknown field key",
args: args{
fieldKey: index.FieldKey{
IndexRuleID: 0,
},
},
},
{
name: "default order",
args: args{
fieldKey: duration,
},
want: []int{50, 200, 500, 1000, 2000},
},
{
name: "invalid range",
args: args{
fieldKey: duration,
termRange: index.NewIntRangeOpts(100, 50, false, false),
},
},
}
preLoadSizes := []int{7, 20, 50}
allTests := make([]struct {
name string
want []int
args args
preloadSize int
}, 0, len(tests)*len(preLoadSizes))
for _, size := range preLoadSizes {
for _, t := range tests {
allTests = append(allTests, struct {
name string
want []int
args args
preloadSize int
}{
name: t.name + " preLoadSize " + fmt.Sprint(size),
want: t.want,
preloadSize: size,
args: t.args,
})
}
}
for _, tt := range allTests {
t.Run(tt.name, func(t *testing.T) {
tester := assert.New(t)
is := require.New(t)
iter, err := store.Iterator(context.TODO(), tt.args.fieldKey, tt.args.termRange, tt.args.orderType, tt.preloadSize)
is.NoError(err)
if iter == nil {
tester.Empty(tt.want)
return
}
defer func() {
tester.NoError(iter.Close())
for i := 0; i < 10; i++ {
is.False(iter.Next())
}
}()
is.NotNil(iter)
var got result
for iter.Next() {
got.items = append(got.items, iter.Val().DocID)
}
for i := 0; i < 10; i++ {
is.False(iter.Next())
}
var wants result
for _, w := range tt.want {
pl := data[w]
wants.items = append(wants.items, pl.ToSlice()...)
}
tester.Equal(wants, got, tt.name)
})
}
}
// SetUpDuration initializes data for testing duration related cases.
func SetUpDuration(t *assert.Assertions, store index.Writer) map[int]posting.List {
r := map[int]posting.List{
50: roaring.NewPostingList(),
200: roaring.NewPostingList(),
500: roaring.NewPostingList(),
1000: roaring.NewPostingList(),
2000: roaring.NewPostingList(),
}
return setUpPartialDuration(t, store, r)
}
func setUpPartialDuration(t *assert.Assertions, store index.Writer, r map[int]posting.List) map[int]posting.List {
idx := make([]int, 0, len(r))
for key := range r {
idx = append(idx, key)
}
sort.Ints(idx)
var batch index.Batch
for i := 100; i < 200; i++ {
id := uint64(i)
for i2, term := range idx {
if i%len(idx) != i2 || r[term] == nil {
continue
}
batch.Documents = append(batch.Documents, index.Document{
Fields: []index.Field{
index.NewIntField(duration, int64(term)),
},
DocID: id,
})
r[term].Insert(id)
}
}
t.NoError(store.Batch(batch))
return r
}