dragonfly/pkg/container/set/safe_set_test.go

383 lines
7.1 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 interface{}
expect func(t *testing.T, ok bool, s Set, value interface{})
}{
{
name: "add value",
value: "foo",
expect: func(t *testing.T, ok bool, s Set, value interface{}) {
assert := assert.New(t)
assert.Equal(ok, true)
assert.Equal(s.Values(), []interface{}{value})
},
},
{
name: "add value failed",
value: "foo",
expect: func(t *testing.T, _ bool, s Set, value interface{}) {
assert := assert.New(t)
ok := s.Add("foo")
assert.Equal(ok, false)
assert.Equal(s.Values(), []interface{}{value})
},
},
}
for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
s := NewSafeSet()
tc.expect(t, s.Add(tc.value), s, tc.value)
})
}
}
func TestSafeSetAdd_Concurrent(t *testing.T) {
runtime.GOMAXPROCS(2)
s := NewSafeSet()
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 interface{}
expect func(t *testing.T, s Set, value interface{})
}{
{
name: "delete value",
value: "foo",
expect: func(t *testing.T, s Set, value interface{}) {
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 Set, _ interface{}) {
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()
s.Add(tc.value)
tc.expect(t, s, tc.value)
})
}
}
func TestSafeSetDelete_Concurrent(t *testing.T) {
runtime.GOMAXPROCS(2)
s := NewSafeSet()
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 interface{}
expect func(t *testing.T, s Set, value interface{})
}{
{
name: "contains value",
value: "foo",
expect: func(t *testing.T, s Set, value interface{}) {
assert := assert.New(t)
assert.Equal(s.Contains(value), true)
},
},
{
name: "contains value does not exist",
value: "foo",
expect: func(t *testing.T, s Set, _ interface{}) {
assert := assert.New(t)
assert.Equal(s.Contains("bar"), false)
},
},
}
for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
s := NewSafeSet()
s.Add(tc.value)
tc.expect(t, s, tc.value)
})
}
}
func TestSafeSetContains_Concurrent(t *testing.T) {
runtime.GOMAXPROCS(2)
s := NewSafeSet()
nums := rand.Perm(N)
interfaces := make([]interface{}, 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 Set)
}{
{
name: "get length",
expect: func(t *testing.T, s Set) {
assert := assert.New(t)
s.Add("foo")
assert.Equal(s.Len(), uint(1))
},
},
{
name: "get empty set length",
expect: func(t *testing.T, s Set) {
assert := assert.New(t)
assert.Equal(s.Len(), uint(0))
},
},
}
for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
s := NewSafeSet()
tc.expect(t, s)
})
}
}
func TestSafeSetLen_Concurrent(t *testing.T) {
runtime.GOMAXPROCS(2)
s := NewSafeSet()
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 Set)
}{
{
name: "get values",
expect: func(t *testing.T, s Set) {
assert := assert.New(t)
s.Add("foo")
assert.Equal(s.Values(), []interface{}{"foo"})
},
},
{
name: "get empty values",
expect: func(t *testing.T, s Set) {
assert := assert.New(t)
assert.Equal(s.Values(), []interface{}(nil))
},
},
{
name: "get multi values",
expect: func(t *testing.T, s Set) {
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()
tc.expect(t, s)
})
}
}
func TestSafeSetValues_Concurrent(t *testing.T) {
runtime.GOMAXPROCS(2)
s := NewSafeSet()
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(rand.Int())
}
wg.Wait()
}
func TestSafeSetRange(t *testing.T) {
tests := []struct {
name string
expect func(t *testing.T, s Set)
}{
{
name: "range values",
expect: func(t *testing.T, s Set) {
assert := assert.New(t)
s.Add("foo")
s.Range(func(v interface{}) bool {
assert.Equal(v, "foo")
return true
})
},
},
{
name: "range values failed",
expect: func(t *testing.T, s Set) {
assert := assert.New(t)
s.Add("foo")
s.Add("bar")
s.Range(func(v interface{}) bool {
assert.Equal(s.Contains(v), true)
return false
})
},
},
}
for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
s := NewSafeSet()
tc.expect(t, s)
})
}
}
func TestSafeSetRange_Concurrent(t *testing.T) {
runtime.GOMAXPROCS(2)
s := NewSafeSet()
var wg sync.WaitGroup
wg.Add(1)
go func() {
elems := s.Values()
i := 0
s.Range(func(v interface{}) bool {
i++
return true
})
if i < len(elems) {
t.Errorf("Values shrunk from %v to %v", elems, i)
}
wg.Done()
}()
for i := 0; i < N; i++ {
s.Add(rand.Int())
}
wg.Wait()
}