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 }