2018-09-28 14:51:47 +00:00
|
|
|
package slices
|
|
|
|
|
|
|
|
import (
|
2018-11-21 14:45:34 +00:00
|
|
|
"bytes"
|
|
|
|
"fmt"
|
2018-09-28 14:51:47 +00:00
|
|
|
"math"
|
|
|
|
"reflect"
|
|
|
|
"testing"
|
|
|
|
"unsafe"
|
2018-11-21 14:45:34 +00:00
|
|
|
|
|
|
|
"github.com/google/go-cmp/cmp"
|
2019-01-08 00:37:16 +00:00
|
|
|
"github.com/influxdata/influxdb/pkg/bytesutil"
|
2018-09-28 14:51:47 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestCopyChunkedByteSlices_oneChunk(t *testing.T) {
|
|
|
|
src := [][]byte{
|
|
|
|
[]byte("influx"),
|
|
|
|
[]byte("data"),
|
|
|
|
}
|
|
|
|
|
|
|
|
dst := CopyChunkedByteSlices(src, 3)
|
|
|
|
if !reflect.DeepEqual(src, dst) {
|
|
|
|
t.Errorf("destination should match source src: %v dst: %v", src, dst)
|
|
|
|
}
|
|
|
|
|
|
|
|
dst[0][1] = 'z'
|
|
|
|
if reflect.DeepEqual(src, dst) {
|
|
|
|
t.Error("destination should not match source")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCopyChunkedByteSlices_multipleChunks(t *testing.T) {
|
|
|
|
src := [][]byte{
|
|
|
|
[]byte("influx"),
|
|
|
|
[]byte("data"),
|
|
|
|
[]byte("is"),
|
|
|
|
[]byte("the"),
|
|
|
|
[]byte("best"),
|
|
|
|
[]byte("time"),
|
|
|
|
[]byte("series"),
|
|
|
|
[]byte("database"),
|
|
|
|
[]byte("in"),
|
|
|
|
[]byte("the"),
|
|
|
|
[]byte("whole"),
|
|
|
|
[]byte("wide"),
|
|
|
|
[]byte("world"),
|
|
|
|
[]byte(":-)"),
|
|
|
|
}
|
|
|
|
|
|
|
|
chunkSize := 4
|
|
|
|
dst := CopyChunkedByteSlices(src, chunkSize)
|
|
|
|
if !reflect.DeepEqual(src, dst) {
|
|
|
|
t.Errorf("destination should match source src: %v dst: %v", src, dst)
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := 0; i < int(math.Ceil(float64(len(src))/float64(chunkSize))); i++ {
|
|
|
|
thisChunkSize := chunkSize
|
|
|
|
if len(src)-thisChunkSize*i < thisChunkSize {
|
|
|
|
thisChunkSize = len(src) - thisChunkSize*i
|
|
|
|
}
|
|
|
|
|
|
|
|
chunk := dst[i*thisChunkSize : (i+1)*thisChunkSize]
|
|
|
|
|
|
|
|
for j := 0; j < thisChunkSize-1; j++ {
|
|
|
|
a := (*reflect.SliceHeader)(unsafe.Pointer(&chunk[j]))
|
|
|
|
b := (*reflect.SliceHeader)(unsafe.Pointer(&chunk[j+1]))
|
|
|
|
if b.Data-a.Data != uintptr(a.Len) {
|
|
|
|
t.Error("chunk elements do not appear to be adjacent, so not part of one chunk")
|
|
|
|
}
|
|
|
|
if a.Cap != a.Len {
|
|
|
|
t.Errorf("slice length != capacity; %d vs %d", a.Len, a.Cap)
|
|
|
|
}
|
|
|
|
if b.Cap != b.Len {
|
|
|
|
t.Errorf("slice length != capacity; %d vs %d", b.Len, b.Cap)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
dst[0][5] = 'z'
|
|
|
|
if reflect.DeepEqual(src, dst) {
|
|
|
|
t.Error("destination should not match source")
|
|
|
|
}
|
|
|
|
}
|
2018-11-21 14:45:34 +00:00
|
|
|
|
2018-11-21 19:00:48 +00:00
|
|
|
const NIL = "<nil>"
|
|
|
|
|
2018-11-21 14:45:34 +00:00
|
|
|
// ss returns a sorted slice of byte slices.
|
|
|
|
func ss(s ...string) [][]byte {
|
|
|
|
r := make([][]byte, len(s))
|
|
|
|
for i := range s {
|
2018-11-21 19:00:48 +00:00
|
|
|
if s[i] != NIL {
|
|
|
|
r[i] = []byte(s[i])
|
|
|
|
}
|
2018-11-21 14:45:34 +00:00
|
|
|
}
|
|
|
|
bytesutil.Sort(r)
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCompareSlice(t *testing.T) {
|
|
|
|
name := func(a, b [][]byte, exp int) string {
|
2018-11-21 19:00:48 +00:00
|
|
|
var as string
|
|
|
|
if a != nil {
|
|
|
|
as = string(bytes.Join(a, nil))
|
|
|
|
} else {
|
|
|
|
as = NIL
|
|
|
|
}
|
|
|
|
var bs string
|
|
|
|
if b != nil {
|
|
|
|
bs = string(bytes.Join(b, nil))
|
|
|
|
} else {
|
|
|
|
bs = NIL
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("%s <=> %s is %d", as, bs, exp)
|
2018-11-21 14:45:34 +00:00
|
|
|
}
|
|
|
|
tests := []struct {
|
|
|
|
a, b [][]byte
|
|
|
|
exp int
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
a: ss("aaa", "bbb", "ccc"),
|
|
|
|
b: ss("aaa", "bbb", "ccc"),
|
|
|
|
exp: 0,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
a: ss("aaa", "bbb", "ccc", "ddd"),
|
|
|
|
b: ss("aaa", "bbb", "ccc"),
|
|
|
|
exp: 1,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
a: ss("aaa", "bbb"),
|
|
|
|
b: ss("aaa", "bbb", "ccc"),
|
|
|
|
exp: -1,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
a: ss("aaa", "bbbb"),
|
|
|
|
b: ss("aaa", "bbb", "ccc"),
|
|
|
|
exp: 1,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
a: ss("aaa", "ccc"),
|
|
|
|
b: ss("aaa", "bbb", "ccc"),
|
|
|
|
exp: 1,
|
|
|
|
},
|
|
|
|
|
2018-11-21 19:00:48 +00:00
|
|
|
{
|
|
|
|
a: ss("aaa", "bbb", NIL),
|
|
|
|
b: ss("aaa", "bbb", "ccc"),
|
|
|
|
exp: -1,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
a: ss("aaa", NIL, "ccc"),
|
|
|
|
b: ss("aaa", NIL, "ccc"),
|
|
|
|
exp: 0,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
a: ss(NIL, "bbb", "ccc"),
|
|
|
|
b: ss("aaa", "bbb", "ccc"),
|
|
|
|
exp: -1,
|
|
|
|
},
|
|
|
|
|
2018-11-21 14:45:34 +00:00
|
|
|
{
|
|
|
|
a: ss("aaa", "aaa"),
|
|
|
|
b: ss("aaa", "bbb", "ccc"),
|
|
|
|
exp: -1,
|
|
|
|
},
|
2018-11-21 19:00:48 +00:00
|
|
|
|
|
|
|
{
|
|
|
|
a: nil,
|
|
|
|
b: ss("aaa", "bbb", "ccc"),
|
|
|
|
exp: -1,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
a: ss("aaa", "bbb"),
|
|
|
|
b: nil,
|
|
|
|
exp: 1,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
a: nil,
|
|
|
|
b: nil,
|
|
|
|
exp: 0,
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
a: [][]byte{},
|
|
|
|
b: nil,
|
|
|
|
exp: 0,
|
|
|
|
},
|
2018-11-21 14:45:34 +00:00
|
|
|
}
|
|
|
|
for _, test := range tests {
|
|
|
|
t.Run(name(test.a, test.b, test.exp), func(t *testing.T) {
|
|
|
|
if got := CompareSlice(test.a, test.b); got != test.exp {
|
|
|
|
t.Errorf("unexpected result, -got/+exp\n%s", cmp.Diff(got, test.exp))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|