2023-03-03 18:15:56 +00:00
|
|
|
// Copyright (c) Tailscale Inc & AUTHORS
|
|
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
|
|
|
|
package slicesx
|
|
|
|
|
|
|
|
import (
|
|
|
|
"reflect"
|
2023-08-17 06:09:53 +01:00
|
|
|
"slices"
|
2023-03-03 18:15:56 +00:00
|
|
|
"testing"
|
2023-09-30 23:28:14 +01:00
|
|
|
|
|
|
|
qt "github.com/frankban/quicktest"
|
2023-03-03 18:15:56 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestInterleave(t *testing.T) {
|
|
|
|
testCases := []struct {
|
|
|
|
name string
|
|
|
|
a, b []int
|
|
|
|
want []int
|
|
|
|
}{
|
|
|
|
{name: "equal", a: []int{1, 3, 5}, b: []int{2, 4, 6}, want: []int{1, 2, 3, 4, 5, 6}},
|
|
|
|
{name: "short_b", a: []int{1, 3, 5}, b: []int{2, 4}, want: []int{1, 2, 3, 4, 5}},
|
|
|
|
{name: "short_a", a: []int{1, 3}, b: []int{2, 4, 6}, want: []int{1, 2, 3, 4, 6}},
|
|
|
|
{name: "len_1", a: []int{1}, b: []int{2, 4, 6}, want: []int{1, 2, 4, 6}},
|
|
|
|
{name: "nil_a", a: nil, b: []int{2, 4, 6}, want: []int{2, 4, 6}},
|
|
|
|
{name: "nil_all", a: nil, b: nil, want: nil},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range testCases {
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
|
|
merged := Interleave(tc.a, tc.b)
|
|
|
|
if !reflect.DeepEqual(merged, tc.want) {
|
|
|
|
t.Errorf("got %v; want %v", merged, tc.want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkInterleave(b *testing.B) {
|
|
|
|
b.ReportAllocs()
|
|
|
|
b.ResetTimer()
|
2024-04-16 21:15:13 +01:00
|
|
|
for range b.N {
|
2023-03-03 18:15:56 +00:00
|
|
|
Interleave(
|
|
|
|
[]int{1, 2, 3},
|
|
|
|
[]int{9, 8, 7},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
2023-06-08 17:22:38 +01:00
|
|
|
|
2023-03-03 18:15:56 +00:00
|
|
|
func TestShuffle(t *testing.T) {
|
|
|
|
var sl []int
|
2024-04-16 21:15:13 +01:00
|
|
|
for i := range 100 {
|
2023-03-03 18:15:56 +00:00
|
|
|
sl = append(sl, i)
|
|
|
|
}
|
|
|
|
|
|
|
|
var wasShuffled bool
|
|
|
|
for try := 0; try < 10; try++ {
|
|
|
|
shuffled := slices.Clone(sl)
|
|
|
|
Shuffle(shuffled)
|
|
|
|
if !reflect.DeepEqual(shuffled, sl) {
|
|
|
|
wasShuffled = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !wasShuffled {
|
|
|
|
t.Errorf("expected shuffle after 10 tries")
|
|
|
|
}
|
|
|
|
}
|
2023-06-08 17:22:38 +01:00
|
|
|
|
|
|
|
func TestPartition(t *testing.T) {
|
|
|
|
var sl []int
|
|
|
|
for i := 1; i <= 10; i++ {
|
|
|
|
sl = append(sl, i)
|
|
|
|
}
|
|
|
|
|
|
|
|
evens, odds := Partition(sl, func(elem int) bool {
|
|
|
|
return elem%2 == 0
|
|
|
|
})
|
|
|
|
|
|
|
|
wantEvens := []int{2, 4, 6, 8, 10}
|
|
|
|
wantOdds := []int{1, 3, 5, 7, 9}
|
|
|
|
if !reflect.DeepEqual(evens, wantEvens) {
|
|
|
|
t.Errorf("evens: got %v, want %v", evens, wantEvens)
|
|
|
|
}
|
|
|
|
if !reflect.DeepEqual(odds, wantOdds) {
|
|
|
|
t.Errorf("odds: got %v, want %v", odds, wantOdds)
|
|
|
|
}
|
|
|
|
}
|
2023-09-30 23:28:14 +01:00
|
|
|
|
|
|
|
func TestEqualSameNil(t *testing.T) {
|
|
|
|
c := qt.New(t)
|
|
|
|
c.Check(EqualSameNil([]string{"a"}, []string{"a"}), qt.Equals, true)
|
|
|
|
c.Check(EqualSameNil([]string{"a"}, []string{"b"}), qt.Equals, false)
|
|
|
|
c.Check(EqualSameNil([]string{"a"}, []string{}), qt.Equals, false)
|
|
|
|
c.Check(EqualSameNil([]string{}, []string{}), qt.Equals, true)
|
|
|
|
c.Check(EqualSameNil(nil, []string{}), qt.Equals, false)
|
|
|
|
c.Check(EqualSameNil([]string{}, nil), qt.Equals, false)
|
|
|
|
c.Check(EqualSameNil[[]string](nil, nil), qt.Equals, true)
|
|
|
|
}
|
2024-01-19 15:51:41 +00:00
|
|
|
|
|
|
|
func TestFilter(t *testing.T) {
|
|
|
|
var sl []int
|
|
|
|
for i := 1; i <= 10; i++ {
|
|
|
|
sl = append(sl, i)
|
|
|
|
}
|
|
|
|
|
|
|
|
evens := Filter(nil, sl, func(elem int) bool {
|
|
|
|
return elem%2 == 0
|
|
|
|
})
|
|
|
|
|
|
|
|
want := []int{2, 4, 6, 8, 10}
|
|
|
|
if !reflect.DeepEqual(evens, want) {
|
|
|
|
t.Errorf("evens: got %v, want %v", evens, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestFilterNoAllocations(t *testing.T) {
|
|
|
|
var sl []int
|
|
|
|
for i := 1; i <= 10; i++ {
|
|
|
|
sl = append(sl, i)
|
|
|
|
}
|
|
|
|
|
|
|
|
want := []int{2, 4, 6, 8, 10}
|
|
|
|
allocs := testing.AllocsPerRun(1000, func() {
|
|
|
|
src := slices.Clone(sl)
|
|
|
|
evens := Filter(src[:0], src, func(elem int) bool {
|
|
|
|
return elem%2 == 0
|
|
|
|
})
|
|
|
|
if !slices.Equal(evens, want) {
|
|
|
|
t.Errorf("evens: got %v, want %v", evens, want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
// 1 alloc for 'src', nothing else
|
|
|
|
if allocs != 1 {
|
|
|
|
t.Fatalf("got %.4f allocs, want 1", allocs)
|
|
|
|
}
|
|
|
|
}
|
2024-05-01 00:36:41 +01:00
|
|
|
|
|
|
|
func TestAppendMatching(t *testing.T) {
|
|
|
|
v := []string{"one", "two", "three", "four"}
|
|
|
|
got := AppendMatching(v[:0], v, func(s string) bool { return len(s) > 3 })
|
|
|
|
|
|
|
|
want := []string{"three", "four"}
|
|
|
|
if !reflect.DeepEqual(got, want) {
|
|
|
|
t.Errorf("got %v; want %v", got, want)
|
|
|
|
}
|
|
|
|
|
|
|
|
wantOrigMem := []string{"three", "four", "three", "four"}
|
|
|
|
if !reflect.DeepEqual(v, wantOrigMem) {
|
|
|
|
t.Errorf("got %v; want %v", v, wantOrigMem)
|
|
|
|
}
|
|
|
|
}
|
2024-07-22 16:36:10 +01:00
|
|
|
|
|
|
|
func TestCutPrefix(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
s, prefix []int
|
|
|
|
after []int
|
|
|
|
found bool
|
|
|
|
}{
|
|
|
|
{"has-prefix", []int{1, 2, 3}, []int{1}, []int{2, 3}, true},
|
|
|
|
{"exact-prefix", []int{1, 2, 3}, []int{1, 2, 3}, []int{}, true},
|
|
|
|
{"blank-prefix", []int{1, 2, 3}, []int{}, []int{1, 2, 3}, true},
|
|
|
|
{"no-prefix", []int{1, 2, 3}, []int{42}, []int{1, 2, 3}, false},
|
|
|
|
{"blank-slice", []int{}, []int{42}, []int{}, false},
|
|
|
|
{"blank-all", []int{}, []int{}, []int{}, true},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
if after, found := CutPrefix(tt.s, tt.prefix); !slices.Equal(after, tt.after) || found != tt.found {
|
|
|
|
t.Errorf("CutPrefix(%v, %v) = %v, %v; want %v, %v", tt.s, tt.prefix, after, found, tt.after, tt.found)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCutSuffix(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
s, suffix []int
|
|
|
|
before []int
|
|
|
|
found bool
|
|
|
|
}{
|
|
|
|
{"has-suffix", []int{1, 2, 3}, []int{3}, []int{1, 2}, true},
|
|
|
|
{"exact-suffix", []int{1, 2, 3}, []int{1, 2, 3}, []int{}, true},
|
|
|
|
{"blank-suffix", []int{1, 2, 3}, []int{}, []int{1, 2, 3}, true},
|
|
|
|
{"no-suffix", []int{1, 2, 3}, []int{42}, []int{1, 2, 3}, false},
|
|
|
|
{"blank-slice", []int{}, []int{42}, []int{}, false},
|
|
|
|
{"blank-all", []int{}, []int{}, []int{}, true},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
if before, found := CutSuffix(tt.s, tt.suffix); !slices.Equal(before, tt.before) || found != tt.found {
|
|
|
|
t.Errorf("CutSuffix(%v, %v) = %v, %v; want %v, %v", tt.s, tt.suffix, before, found, tt.before, tt.found)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2024-09-12 00:32:05 +01:00
|
|
|
|
|
|
|
func TestFirstLastEqual(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
in string
|
|
|
|
v byte
|
|
|
|
f func([]byte, byte) bool
|
|
|
|
want bool
|
|
|
|
}{
|
|
|
|
{"first-empty", "", 'f', FirstEqual[byte], false},
|
|
|
|
{"first-true", "foo", 'f', FirstEqual[byte], true},
|
|
|
|
{"first-false", "foo", 'b', FirstEqual[byte], false},
|
|
|
|
{"last-empty", "", 'f', LastEqual[byte], false},
|
|
|
|
{"last-true", "bar", 'r', LastEqual[byte], true},
|
|
|
|
{"last-false", "bar", 'o', LastEqual[byte], false},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
if got := tt.f([]byte(tt.in), tt.v); got != tt.want {
|
|
|
|
t.Errorf("got %v; want %v", got, tt.want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|