mirror of
https://github.com/tailscale/tailscale.git
synced 2025-01-10 01:53:49 +00:00
d500a92926
The standard library includes these for strings and byte slices, but it lacks similar functions for generic slices of comparable types. Although they are not as commonly used, these functions are useful in scenarios such as working with field index sequences (i.e., []int) via reflection. Updates #12687 Signed-off-by: Nick Khyl <nickk@tailscale.com>
200 lines
5.1 KiB
Go
200 lines
5.1 KiB
Go
// Copyright (c) Tailscale Inc & AUTHORS
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
package slicesx
|
|
|
|
import (
|
|
"reflect"
|
|
"slices"
|
|
"testing"
|
|
|
|
qt "github.com/frankban/quicktest"
|
|
)
|
|
|
|
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()
|
|
for range b.N {
|
|
Interleave(
|
|
[]int{1, 2, 3},
|
|
[]int{9, 8, 7},
|
|
)
|
|
}
|
|
}
|
|
|
|
func TestShuffle(t *testing.T) {
|
|
var sl []int
|
|
for i := range 100 {
|
|
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")
|
|
}
|
|
}
|
|
|
|
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)
|
|
}
|
|
}
|
|
|
|
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)
|
|
}
|
|
|
|
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)
|
|
}
|
|
}
|
|
|
|
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)
|
|
}
|
|
}
|
|
|
|
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)
|
|
}
|
|
})
|
|
}
|
|
}
|