mirror of
https://github.com/dolthub/dolt.git
synced 2026-01-23 10:27:34 -06:00
241 lines
5.4 KiB
Go
241 lines
5.4 KiB
Go
// Copyright 2022 Dolthub, 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 prolly
|
|
|
|
import (
|
|
"context"
|
|
"encoding/binary"
|
|
"io"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/dolthub/dolt/go/store/prolly/tree"
|
|
"github.com/dolthub/dolt/go/store/val"
|
|
)
|
|
|
|
var rangeTuples = []val.Tuple{
|
|
intNullTuple(nil, &nine), // 0
|
|
intTuple(1, 1), // 1
|
|
intTuple(1, 2), // 2
|
|
intTuple(1, 3), // 3
|
|
intTuple(2, 1), // 4
|
|
intTuple(2, 2), // 5
|
|
intTuple(2, 3), // 6
|
|
intTuple(3, 1), // 7
|
|
intTuple(3, 2), // 8
|
|
intTuple(3, 3), // 9
|
|
intTuple(4, 1), // 10
|
|
intTuple(4, 2), // 11
|
|
intTuple(4, 3), // 12
|
|
intNullTuple(&nine, nil), // 13
|
|
}
|
|
|
|
var nine = int32(9)
|
|
|
|
func TestRangeSearch(t *testing.T) {
|
|
intType := val.Type{
|
|
Enc: val.Int32Enc,
|
|
Nullable: true,
|
|
}
|
|
twoCol := val.NewTupleDescriptor(
|
|
intType, // c0
|
|
intType, // c1
|
|
)
|
|
|
|
tests := []struct {
|
|
name string
|
|
testRange Range
|
|
physical [2]int // physical range scan
|
|
logical []int // logical range scan
|
|
}{
|
|
{
|
|
name: "unbound range",
|
|
testRange: Range{
|
|
Fields: []RangeField{
|
|
{
|
|
Lo: Bound{Binding: false},
|
|
Hi: Bound{Binding: false},
|
|
},
|
|
},
|
|
Desc: twoCol,
|
|
},
|
|
physical: [2]int{0, 14},
|
|
logical: []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14},
|
|
},
|
|
|
|
// first column ranges
|
|
{
|
|
name: "c0 > 1",
|
|
testRange: Range{
|
|
Fields: []RangeField{
|
|
{
|
|
Lo: Bound{Binding: true, Inclusive: false, Value: intVal(1)},
|
|
},
|
|
},
|
|
Desc: twoCol,
|
|
},
|
|
physical: [2]int{4, 14},
|
|
logical: []int{4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14},
|
|
},
|
|
{
|
|
name: "c0 < 1",
|
|
testRange: Range{
|
|
Fields: []RangeField{
|
|
{
|
|
Lo: Bound{Binding: true, Inclusive: false, Value: nil},
|
|
Hi: Bound{Binding: true, Inclusive: false, Value: intVal(1)},
|
|
},
|
|
},
|
|
Desc: twoCol,
|
|
},
|
|
physical: [2]int{1, 1},
|
|
logical: []int{},
|
|
},
|
|
{
|
|
name: "2 <= c0 <= 3",
|
|
testRange: Range{
|
|
Fields: []RangeField{
|
|
{
|
|
Lo: Bound{Binding: true, Inclusive: true, Value: intVal(2)},
|
|
Hi: Bound{Binding: true, Inclusive: true, Value: intVal(3)},
|
|
},
|
|
},
|
|
Desc: twoCol,
|
|
},
|
|
physical: [2]int{4, 10},
|
|
logical: []int{4, 5, 6, 7, 8, 9, 10, 11},
|
|
},
|
|
{
|
|
name: "c0 = NULL",
|
|
testRange: Range{
|
|
Fields: []RangeField{
|
|
{
|
|
Lo: Bound{Binding: true, Inclusive: true, Value: nil},
|
|
Hi: Bound{Binding: true, Inclusive: true, Value: nil},
|
|
Exact: true,
|
|
},
|
|
},
|
|
Desc: twoCol,
|
|
},
|
|
physical: [2]int{0, 1},
|
|
logical: []int{0},
|
|
},
|
|
|
|
// second column ranges
|
|
{
|
|
name: "c1 == 2",
|
|
testRange: Range{
|
|
Fields: []RangeField{
|
|
{
|
|
Lo: Bound{Binding: true, Inclusive: false, Value: nil},
|
|
},
|
|
{
|
|
Lo: Bound{Binding: true, Inclusive: true, Value: intVal(2)},
|
|
Hi: Bound{Binding: true, Inclusive: true, Value: intVal(2)},
|
|
Exact: true,
|
|
},
|
|
},
|
|
Desc: twoCol,
|
|
},
|
|
physical: [2]int{1, 14},
|
|
logical: []int{2, 5, 8, 11},
|
|
},
|
|
}
|
|
|
|
values := make([]val.Tuple, len(rangeTuples))
|
|
for i := range values {
|
|
values[i] = make(val.Tuple, 2)
|
|
}
|
|
testNode := tree.NewTupleLeafNode(rangeTuples, values)
|
|
tm := NewMap(testNode, nil, twoCol, val.TupleDesc{})
|
|
|
|
for _, test := range tests {
|
|
t.Run(test.name, func(t *testing.T) {
|
|
ctx := context.Background()
|
|
rng := test.testRange
|
|
lo, hi := test.physical[0], test.physical[1]
|
|
|
|
startSearch := rangeStartSearchFn(rng)
|
|
idx := startSearch(testNode)
|
|
assert.Equal(t, lo, idx, "range should start at index %d", lo)
|
|
|
|
stopSearch := rangeStopSearchFn(rng)
|
|
idx = stopSearch(testNode)
|
|
assert.Equal(t, hi, idx, "range should stop before index %d", hi)
|
|
|
|
// validate physical range (unfiltered iter)
|
|
iter, err := treeIterFromRange(ctx, testNode, nil, test.testRange)
|
|
require.NoError(t, err)
|
|
expected := rangeTuples[lo:hi]
|
|
|
|
i := 0
|
|
for {
|
|
act, _, err := iter.Next(ctx)
|
|
if err == io.EOF {
|
|
break
|
|
}
|
|
require.NoError(t, err)
|
|
require.True(t, i < len(expected))
|
|
assert.Equal(t, expected[i], act)
|
|
i++
|
|
}
|
|
|
|
// validate logical range
|
|
iter2, err := tm.IterRange(ctx, rng)
|
|
require.NoError(t, err)
|
|
expected2 := test.logical
|
|
|
|
i = 0
|
|
for {
|
|
act, _, err := iter2.Next(ctx)
|
|
if err == io.EOF {
|
|
break
|
|
}
|
|
exp := rangeTuples[expected2[i]]
|
|
require.NoError(t, err)
|
|
assert.Equal(t, exp, act)
|
|
i++
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func intVal(i int32) (buf []byte) {
|
|
buf = make([]byte, 4)
|
|
binary.LittleEndian.PutUint32(buf, uint32(i))
|
|
return
|
|
}
|
|
|
|
func intNullTuple(ints ...*int32) val.Tuple {
|
|
types := make([]val.Type, len(ints))
|
|
for i := range types {
|
|
types[i] = val.Type{
|
|
Enc: val.Int32Enc,
|
|
Nullable: true,
|
|
}
|
|
}
|
|
|
|
desc := val.NewTupleDescriptor(types...)
|
|
tb := val.NewTupleBuilder(desc)
|
|
for i, val := range ints {
|
|
if val != nil {
|
|
tb.PutInt32(i, *val)
|
|
}
|
|
}
|
|
return tb.Build(sharedPool)
|
|
}
|