dragonfly/pkg/container/set/safe_set_test.go

377 lines
7.2 KiB
Go

/*
* Copyright 2020 The Dragonfly Authors
*
* 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 set
import (
"math/rand"
"runtime"
"sync"
"testing"
"github.com/stretchr/testify/assert"
)
const N = 1000
func TestSafeSetAdd(t *testing.T) {
tests := []struct {
name string
value string
expect func(t *testing.T, ok bool, s SafeSet[string], value string)
}{
{
name: "add value",
value: "foo",
expect: func(t *testing.T, ok bool, s SafeSet[string], value string) {
assert := assert.New(t)
assert.Equal(ok, true)
assert.Equal(s.Values(), []string{value})
},
},
{
name: "add value failed",
value: "foo",
expect: func(t *testing.T, _ bool, s SafeSet[string], value string) {
assert := assert.New(t)
ok := s.Add("foo")
assert.Equal(ok, false)
assert.Equal(s.Values(), []string{value})
},
},
}
for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
s := NewSafeSet[string]()
tc.expect(t, s.Add(tc.value), s, tc.value)
})
}
}
func TestSafeSetAdd_Concurrent(t *testing.T) {
runtime.GOMAXPROCS(2)
s := NewSafeSet[int]()
nums := rand.Perm(N)
var wg sync.WaitGroup
wg.Add(len(nums))
for i := 0; i < len(nums); i++ {
go func(i int) {
s.Add(i)
wg.Done()
}(i)
}
wg.Wait()
for _, n := range nums {
if !s.Contains(n) {
t.Errorf("Set is missing element: %v", n)
}
}
}
func TestSafeSetDelete(t *testing.T) {
tests := []struct {
name string
value string
expect func(t *testing.T, s SafeSet[string], value string)
}{
{
name: "delete value",
value: "foo",
expect: func(t *testing.T, s SafeSet[string], value string) {
assert := assert.New(t)
s.Delete(value)
assert.Equal(s.Len(), uint(0))
},
},
{
name: "delete value does not exist",
value: "foo",
expect: func(t *testing.T, s SafeSet[string], _ string) {
assert := assert.New(t)
s.Delete("bar")
assert.Equal(s.Len(), uint(1))
},
},
}
for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
s := NewSafeSet[string]()
s.Add(tc.value)
tc.expect(t, s, tc.value)
})
}
}
func TestSafeSetDelete_Concurrent(t *testing.T) {
runtime.GOMAXPROCS(2)
s := NewSafeSet[int]()
nums := rand.Perm(N)
for _, v := range nums {
s.Add(v)
}
var wg sync.WaitGroup
wg.Add(len(nums))
for _, v := range nums {
go func(i int) {
s.Delete(i)
wg.Done()
}(v)
}
wg.Wait()
if s.Len() != 0 {
t.Errorf("Expected len 0; got %v", s.Len())
}
}
func TestSafeSetContains(t *testing.T) {
tests := []struct {
name string
value string
expect func(t *testing.T, s SafeSet[string], value string)
}{
{
name: "contains value",
value: "foo",
expect: func(t *testing.T, s SafeSet[string], value string) {
assert := assert.New(t)
assert.Equal(s.Contains(string(value)), true)
},
},
{
name: "contains value does not exist",
value: "foo",
expect: func(t *testing.T, s SafeSet[string], _ string) {
assert := assert.New(t)
assert.Equal(s.Contains("bar"), false)
},
},
}
for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
s := NewSafeSet[string]()
s.Add(tc.value)
tc.expect(t, s, tc.value)
})
}
}
func TestSafeSetContains_Concurrent(t *testing.T) {
runtime.GOMAXPROCS(2)
s := NewSafeSet[int]()
nums := rand.Perm(N)
interfaces := make([]int, 0)
for _, v := range nums {
s.Add(v)
interfaces = append(interfaces, v)
}
var wg sync.WaitGroup
for range nums {
wg.Add(1)
go func() {
s.Contains(interfaces...)
wg.Done()
}()
}
wg.Wait()
}
func TestSetSafeLen(t *testing.T) {
tests := []struct {
name string
expect func(t *testing.T, s SafeSet[string])
}{
{
name: "get length",
expect: func(t *testing.T, s SafeSet[string]) {
assert := assert.New(t)
s.Add("foo")
assert.Equal(s.Len(), uint(1))
},
},
{
name: "get empty set length",
expect: func(t *testing.T, s SafeSet[string]) {
assert := assert.New(t)
assert.Equal(s.Len(), uint(0))
},
},
}
for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
s := NewSafeSet[string]()
tc.expect(t, s)
})
}
}
func TestSafeSetLen_Concurrent(t *testing.T) {
runtime.GOMAXPROCS(2)
s := NewSafeSet[int]()
var wg sync.WaitGroup
wg.Add(1)
go func() {
elems := s.Len()
for i := 0; i < N; i++ {
newElems := s.Len()
if newElems < elems {
t.Errorf("Len shrunk from %v to %v", elems, newElems)
}
}
wg.Done()
}()
for i := 0; i < N; i++ {
s.Add(rand.Int())
}
wg.Wait()
}
func TestSafeSetValues(t *testing.T) {
tests := []struct {
name string
expect func(t *testing.T, s SafeSet[string])
}{
{
name: "get values",
expect: func(t *testing.T, s SafeSet[string]) {
assert := assert.New(t)
s.Add("foo")
assert.Equal(s.Values(), []string{"foo"})
},
},
{
name: "get empty values",
expect: func(t *testing.T, s SafeSet[string]) {
assert := assert.New(t)
assert.Equal(s.Values(), []string(nil))
},
},
{
name: "get multi values",
expect: func(t *testing.T, s SafeSet[string]) {
assert := assert.New(t)
s.Add("foo")
s.Add("bar")
assert.Contains(s.Values(), "bar")
assert.Contains(s.Values(), "foo")
},
},
}
for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
s := NewSafeSet[string]()
tc.expect(t, s)
})
}
}
func TestSafeSetValues_Concurrent(t *testing.T) {
runtime.GOMAXPROCS(2)
s := NewSafeSet[int]()
var wg sync.WaitGroup
wg.Add(1)
go func() {
elems := s.Values()
for i := 0; i < N; i++ {
newElems := s.Values()
if len(newElems) < len(elems) {
t.Errorf("Values shrunk from %v to %v", elems, newElems)
}
}
wg.Done()
}()
for i := 0; i < N; i++ {
s.Add(i)
}
wg.Wait()
}
func TestSafeSetClear(t *testing.T) {
tests := []struct {
name string
expect func(t *testing.T, s SafeSet[string])
}{
{
name: "clear empty set",
expect: func(t *testing.T, s SafeSet[string]) {
assert := assert.New(t)
s.Clear()
assert.Equal(s.Values(), []string(nil))
},
},
{
name: "clear set",
expect: func(t *testing.T, s SafeSet[string]) {
assert := assert.New(t)
assert.Equal(s.Add("foo"), true)
s.Clear()
assert.Equal(s.Values(), []string(nil))
assert.Equal(s.Add("foo"), true)
assert.Equal(s.Values(), []string{"foo"})
},
},
}
for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
s := NewSafeSet[string]()
tc.expect(t, s)
})
}
}
func TestSafeSetClear_Concurrent(t *testing.T) {
runtime.GOMAXPROCS(2)
s := NewSafeSet[int]()
nums := rand.Perm(N)
var wg sync.WaitGroup
wg.Add(len(nums))
for i := 0; i < len(nums); i++ {
go func(i int) {
s.Add(i)
s.Clear()
wg.Done()
}(i)
}
wg.Wait()
for _, n := range nums {
if s.Contains(n) {
t.Errorf("SafeSet contains element: %v", n)
}
}
}